Java面试题-Java核心基础-第十三天(序列化)

目录

一、Java序列化与反序列化是什么?

二、为什么需要序列化与反序列化?

三、序列化的实现方式有哪些?

四、什么是serialVersionUID?

五、为什么还要显示指定serialVersionUID

六、serialVersionUID什么时候修改?

七、Java序列化如果有些字段不想序列化,怎么办?

八、静态变量会序列化吗?


一、Java序列化与反序列化是什么?

Java序列化就是将对象转为字节序列

Java反序列化就是将字节序列转为内存中的对象

二、为什么需要序列化与反序列化?

其实大的来说就是为了保存对象的信息 状态信息 因为JVM一旦关闭 其对象也会被销毁 如何持久化对象就需要使用到序列化机制

另外就是为了便于网络传输传输信息,传输字节序列给对方,对方接收到再进行反序列化就能获取到对象信息,像RPC或者是OpenFeign远程调用就是将对象进行序列化与反序列化的过程(其实就是实现分布式对象,多个服务可以使用同一对象)

除此之外,序列化与反序列化机制还能用于实现深拷贝,序列化不仅只序列化当前对象,其对象里面的里面也会递归的序列化

三、序列化的实现方式有哪些?

类实现Serializable接口 或者是实现Externalizable接口

实现Serializable接口的方式很简单,只需要实现这个接口,不需要实现里面的任何方法,因为里面就没方法,然后在类里面指定一个serialVersionUID就行 注意这个必须是static final long的

而实现Externalizable 接口的方式,就更加的灵活了,可以指定我们需要序列化的字段,而如何指定呢?就通过实现Externalizable接口接口中的readExternal和writeExternal方法就可以了

例子:

java 复制代码
public class User implements Externalizable {

   private String name;
   private int age;

   public String getName() {
       return name;
   }
   public void setName(String name) {
       this.name = name;
   }
   public void writeExternal(ObjectOutput out) throws IOException {
       out.writeObject(name);
   }
   public void readExternal(ObjectInput in) throws IOException, ClassNotFoundException {
       name = (String) in.readObject();
   }

   @Override
   public String toString() {
       return "User{" +
               "name='" + name +
               '}';
   }
}

两者的对比:

Serializable接口更为简单,它是按照系统默认的序列化规则,能序列化所有属性。但是效率不高

Externalizable接口需要自己重写两个方法,可以指定序列化字段,如果不重写的话,默认是属性类型的默认值 效率高

四、什么是serialVersionUID?

就是一个类的标识,需要依靠这个来反序列化。它的过程是:拿到序列化中的serialVersionUID值,然后与本地的类的serialVersionUID作比较,如果相同则反序列化成本地类对象,如果不同则序列化失败

五、为什么还要显示指定serialVersionUID

如果不显示指定,那么这个值就会根据类的属性来自动生成有一个serialVersionUID值,因此如果类中属性变化了,这个值就会发生变化。

如果再我们序列化之后修改了类中的属性,那么此时这个serialVersionUID值就会与它原先当初序列化时类中的serialVersionUID不一样,所以就会反序列化失败。

因此一般我们要手动的指定一个serialVersionUID值

六、serialVersionUID什么时候修改?

一般的时候不要修改,除非软件不兼容的情况 因为不兼容 所以不允许类随意发生改变

七、Java序列化如果有些字段不想序列化,怎么办?

使用transient修饰 或者类实现Externalizable接口的方式重写方法指定

八、静态变量会序列化吗?

不会,因为静态变量属于类,而序列化的是对象的信息。

至于为什么静态的serialVersionUID可以被序列化 是因为JVM作了特殊判断,因为需要依靠它来进行反序列化

相关推荐
无心水1 小时前
【分布式利器:RocketMQ】2、RocketMQ消息重复?3种幂等方案,彻底解决重复消费(附代码实操)
网络·数据库·rocketmq·java面试·消息幂等·重复消费·分布式利器
陈果然DeepVersion5 天前
Java大厂面试真题:从Spring Boot到AI微服务的三轮技术拷问
spring boot·redis·微服务·ai·智能客服·java面试·rag
陈果然DeepVersion5 天前
Java大厂面试真题:从Spring Boot到AI微服务的三轮技术拷问(二)
spring boot·redis·spring cloud·微服务·ai·java面试·rag
cherry523010 天前
Java大厂面试真题:Spring Boot + 微服务 + 缓存架构三轮技术拷问实录
jvm·spring boot·mysql·微服务·java面试·分布式架构·redis缓存
Java爱好狂.11 天前
分布式ID|从源码角度深度解析美团Leaf双Buffer优化方案
java·数据库·分布式·分布式id·es·java面试·java程序员
陈果然DeepVersion12 天前
Java大厂面试真题:Spring Boot+微服务+AI智能客服三轮技术拷问实录(四)
spring boot·redis·微服务·kafka·spring security·智能客服·java面试
Javatutouhouduan14 天前
记一次redis主从切换导致的数据丢失与陷入只读状态故障
java·redis·设计模式·java面试·高可用·java后端·java程序员
陈果然DeepVersion14 天前
Java大厂面试真题:Spring Boot+Kafka+AI智能客服场景全流程解析(六)
spring boot·kafka·消息队列·向量数据库·java面试·rag·ai智能客服