深入探索Java对象的序列化

本文涉及的产品
系统运维管理,不限时长
简介:
对象序列化就是把对象写入到输出流中,用来存储或者传输。
对象的反序列化就是从输入流中读取对象。
要序列化的对象应该实现Serializable接口。
Serializable接口是一个标识接口,没有抽象方法。
Serializable有一个子接口Externalizable,实现Externalizable接口的类可以自行控制对象序列化荷反序列化过程。
一般来说,没有必要自己实现序列化接口,直接交给Java虚拟机是上策。
实现了序列化接口的类,如果其成员不需要序列化进去,则使用transient关键字进行修饰。
 
下面给出个例子:
import java.io.*; 

/** 
* Java对象的序列化测试 
* File: ObjectStreamTest.java 
* User: leizhimin 
* Date: 2008-3-12 20:41:43 
*/
 
public  class ObjectStreamTest { 
     public  static  void main(String args[]) { 
        testObjectSeri(); 
        testObjectInSeri(); 

    } 

     /** 
     * 对象序列化测试 
     */
 
     public  static  void testObjectSeri() { 
        Person person =  new Person( "熔岩""341022225562156""lavasoft"); 
        FileOutputStream fos =  null
        ObjectOutputStream oos =  null
         try { 
            fos =  new FileOutputStream( "Q:\\study\\java5study\\src\\io\\person.dat"); 
            oos =  new ObjectOutputStream(fos); 
            oos.writeObject(person); 
        }  catch (FileNotFoundException e) { 
            System.out.println( "找不到指定的文件!"); 
            e.printStackTrace(); 
        }  catch (IOException e) { 
            e.printStackTrace(); 
        }  finally { 
             try { 
                oos.flush(); 
                oos.close(); 
            }  catch (IOException e) { 
                e.printStackTrace(); 
            } 
        } 
    } 

     /** 
     * 对象反序列化测试 
     */
 
     public  static  void testObjectInSeri() { 
        FileInputStream fis =  null
        ObjectInputStream ois =  null
        Person person =  null
         try { 
            fis =  new FileInputStream( "Q:\\study\\java5study\\src\\io\\person.dat"); 
            ois =  new ObjectInputStream(fis); 
            person = (Person) ois.readObject(); 
        }  catch (FileNotFoundException e) { 
            e.printStackTrace(); 
        }  catch (IOException e) { 
            e.printStackTrace(); 
        }  catch (ClassNotFoundException e) { 
            e.printStackTrace(); 
        }  finally { 
             try { 
                ois.close(); 
            }  catch (IOException e) { 
                e.printStackTrace(); 
            } 
        } 

        System.out.println(person.toString()); 
    } 


/** 
* 测试序列化所用的类 
*/
 
class Person  implements Serializable { 
     private String username; 
     private String cardNumber; 
     private  transient String password; 

     public Person(String username, String cardNumber, String password) { 
         this.username = username; 
         this.cardNumber = cardNumber; 
         this.password = password; 
    } 

     public String getUsername() { 
         return username; 
    } 

     public  void setUsername(String username) { 
         this.username = username; 
    } 

     public String getCardNumber() { 
         return cardNumber; 
    } 

     public  void setCardNumber(String cardNumber) { 
         this.cardNumber = cardNumber; 
    } 

     public String getPassword() { 
         return password; 
    } 

     public  void setPassword(String password) { 
         this.password = password; 
    } 

     public String toString() { 
        StringBuffer sb =  new StringBuffer( this.getClass().getName()); 
        sb.append( "["); 
        sb.append( "\n\t"); 
        sb.append( "username=" +  this.username); 
        sb.append( "\n\t"); 
        sb.append( "cardNumber=" +  this.cardNumber); 
        sb.append( "\n\t"); 
        sb.append( "password=" +  this.password); 
        sb.append( "]"); 
         return sb.toString(); 
    } 
}
 
运行结果为:
io.Person[ 
    username=熔岩 
    cardNumber=341022225562156 
    password= null

Process finished with exit code 0 
 
属性password=null,说明在序列化过程中忽略了。
 
说到此,还有一个容易忽略的问题--serialVersionUID :
 
序列化运行时使用一个称为 serialVersionUID 的版本号与每个可序列化类相关联,该序列号在反序列化过程中用于验证序列化对象的发送者和接收者是否为该对象加载了与序列化兼容的类。如果接收者加载的该对象的类的 serialVersionUID 与对应的发送者的类的版本号不同,则反序列化将会导致  InvalidClassException 。可序列化类可以通过声明名为  "serialVersionUID"  的字段(该字段必须是静态 (static)、最终 (final) 的  long  型字段)显式声明其自己的 serialVersionUID:
 ANY-ACCESS-MODIFIER static final long serialVersionUID = 42L;
 
如果可序列化类未显式声明 serialVersionUID,则序列化运行时将基于该类的各个方面计算该类的默认 serialVersionUID 值,如“Java(TM) 对象序列化规范”中所述。不过, 强烈建议 所有可序列化类都显式声明 serialVersionUID 值,原因计算默认的 serialVersionUID 对类的详细信息具有较高的敏感性,根据编译器实现的不同可能千差万别,这样在反序列化过程中可能会导致意外的  InvalidClassException。因此,为保证 serialVersionUID 值跨不同 java 编译器实现的一致性,序列化类必须声明一个明确的 serialVersionUID 值。还强烈建议使用  private 修改器显示声明 serialVersionUID(如果可能),原因是这种声明仅应用于立即声明类 -- serialVersionUID 字段作为继承成员没有用处。
 
serialVersionUID 在Eclipse里可以自动生成,可是在其他大部分IDE工具里面都不能自动生成。但是这个long型值取多少,心里没底,与其写还不如不写。
 

本文转自 leizhimin 51CTO博客,原文链接:http://blog.51cto.com/lavasoft/65542,如需转载请自行联系原作者
相关文章
|
25天前
|
存储 Java
java的对象详解
在Java中,对象是根据类模板实例化的内存实体,具有唯一标识符、属性及行为。通过`new`关键字实例化对象并用构造方法初始化。变量存储的是对象引用而非对象本身,属性描述对象状态,方法定义其行为。Java利用垃圾回收机制自动处理不再使用的对象内存回收,极大地简化了对象生命周期管理,同时对象具备封装、继承和多态性,促进了代码的重用与模块化设计。这使得Java程序更易于理解、维护和扩展。
|
9天前
|
Java
java基础(12)抽象类以及抽象方法abstract以及ArrayList对象使用
本文介绍了Java中抽象类和抽象方法的使用,以及ArrayList的基本操作,包括添加、获取、删除元素和判断列表是否为空。
13 2
java基础(12)抽象类以及抽象方法abstract以及ArrayList对象使用
|
20天前
|
Java 编译器
Java——类与对象(继承和多态)
本文介绍了面向对象编程中的继承概念,包括如何避免重复代码、构造方法的调用规则、成员变量的访问以及权限修饰符的使用。文中详细解释了继承与组合的区别,并探讨了多态的概念,包括向上转型、向下转型和方法的重写。此外,还讨论了静态绑定和动态绑定的区别,以及多态带来的优势和弊端。
22 9
Java——类与对象(继承和多态)
|
20天前
|
SQL Java 编译器
Java——类与对象(封装)
封装是面向对象编程中的概念,指将数据(属性)和相关操作(方法)组合成独立单元(类),使外部无法直接访问对象的内部状态,只能通过提供的方法进行交互,从而保护数据安全。例如,手机将各种组件封装起来,只暴露必要的接口供外部使用。实现封装时,使用`private`关键字修饰成员变量,并提供`get`和`set`方法进行访问和修改。此外,介绍了包的概念、导入包的方式及其注意事项,以及`static`关键字的使用,包括静态变量和方法的初始化与代码块的加载顺序。
25 10
Java——类与对象(封装)
|
20天前
|
Java C语言
Java——类与对象
这段内容介绍了Java中的类和对象、`this`关键字及构造方法的基本概念。类是对现实世界事物的抽象描述,包含属性和方法;对象是类的实例,通过`new`关键字创建。`this`关键字用于区分成员变量和局部变量,构造方法用于初始化对象。此外,还介绍了标准JavaBean的要求和生成方法。
21 9
Java——类与对象
|
10天前
|
存储 Java
Java编程中的对象和类
【8月更文挑战第55天】在Java的世界中,“对象”与“类”是构建一切的基础。就像乐高积木一样,类定义了形状和结构,而对象则是根据这些设计拼装出来的具体作品。本篇文章将通过一个简单的例子,展示如何从零开始创建一个类,并利用它来制作我们的第一个Java对象。准备好让你的编程之旅起飞了吗?让我们一起来探索这个神奇的过程!
25 10
|
7天前
|
JSON NoSQL Java
redis的java客户端的使用(Jedis、SpringDataRedis、SpringBoot整合redis、redisTemplate序列化及stringRedisTemplate序列化)
这篇文章介绍了在Java中使用Redis客户端的几种方法,包括Jedis、SpringDataRedis和SpringBoot整合Redis的操作。文章详细解释了Jedis的基本使用步骤,Jedis连接池的创建和使用,以及在SpringBoot项目中如何配置和使用RedisTemplate和StringRedisTemplate。此外,还探讨了RedisTemplate序列化的两种实践方案,包括默认的JDK序列化和自定义的JSON序列化,以及StringRedisTemplate的使用,它要求键和值都必须是String类型。
redis的java客户端的使用(Jedis、SpringDataRedis、SpringBoot整合redis、redisTemplate序列化及stringRedisTemplate序列化)
|
22天前
|
存储 Java
Java的对象和类的相同之处和不同之处
在 Java 中,对象和类是面向对象编程的核心。
|
13天前
|
存储 Java 开发者
Java编程中的对象序列化与反序列化
【9月更文挑战第20天】在本文中,我们将探索Java编程中的一个核心概念——对象序列化与反序列化。通过简单易懂的语言和直观的代码示例,你将学会如何将对象状态保存为字节流,以及如何从字节流恢复对象状态。这不仅有助于理解Java中的I/O机制,还能提升你的数据持久化能力。准备好让你的Java技能更上一层楼了吗?让我们开始吧!
|
15天前
|
Java
Java实现:将带时区的时间字符串转换为LocalDateTime对象
通过上述方法,你可以将带时区的时间字符串准确地转换为 `LocalDateTime`对象,这对于处理不需要时区信息的日期和时间场景非常有用。
205 4
下一篇
无影云桌面