- 浏览: 375185 次
- 性别:
- 来自: 大连
文章分类
最新评论
-
Kitty19872010:
在书《Principles of Concurrent and ...
Java中的volatile关键字 -
奇林醉:
受用了!
JAVA Clone机制 -
guji528:
对result type做了详细描述,好。
struts2 strus.xml中result类型及含义 -
water84222:
dc444 写道为了回帖还做了个测验,真麻烦。
楼主对vol ...
Java中的volatile关键字 -
dc444:
为了回帖还做了个测验,真麻烦。
楼主对volatile是有一 ...
Java中的volatile关键字
1.什么是序列化
对象的寿命通常随着生成该对象的程序的终止而终止。有时候,可
能需要
将
对象的状态保存下来,在需要时再将对象恢复。我们把对象的这种能记录自己的状态以便将来再生的能力
。
叫作对象的持续性(persistence)。对象通过写出描述自己状态的数值来记录自己 ,这个过程叫对象的序列化(Serialization)
。序列化的主要任务是写出对象实例变量的数值。如果交量是另一对象的引用,则引用的对象也要序列化。这个过程是递归的,序列化可能要涉及一个复杂树结构的
单行化,包括原有对象、对象的对象、对象的对象的对象等等。对象所有权的层次结构称为图表(graph)。
当两个进程在进行远程通信时,彼此可以发送各种类型的数据。无论是何种类型的数据,都会以二进制序列的形式在网络上传送。发送方需要把这个Java对象转换为字节序列,才能在网络上传送;接收方则需要把字节序列再恢复为Java对象。
把Java对象转换为字节序列的过程称为对象的序列化。
把字节序列恢复为Java对象的过程称为对象的反序列化
对象的序列化主要有两种用途:
1) 把对象的字节序列永久地保存到硬盘上,通常存放在一个文件中;
2) 在网络上传送对象的字节序列。
java.io.ObjectOutputStream代表对象输出流,它的writeObject(Object obj)方法可对参数指定的obj对象进行序列化,把得到的字节序列写到一个目标输出流中。
java.io.ObjectInputStream代表对象输入流,它的readObject()方法从一个源输入流中读取字节序列,再把它们反序列化为一个对象,并将其返回。、
只有实现了Serializable和Externalizable接口的类的对象才能被序列化。Externalizable接口继承自
Serializable接口,实现Externalizable接口的类完全由自身来控制序列化的行为,而仅实现Serializable接口的类可以
采用默认的序列化方式 。Serializable接口中没有任何的方法。当一个类声明要实现Serializable接口时,只是表明该类参加
串行化协议,而不需要实现任何特殊的方法。
对象序列化包括如下步骤:
1) 创建一个对象输出流,它可以包装一个其他类型的目标输出流,如文件输出流;
2) 通过对象输出流的writeObject()方法写对象。
对象反序列化的步骤如下:
1) 创建一个对象输入流,它可以包装一个其他类型的源输入流,如文件输入流;
2) 通过对象输入流的readObject()方法读取对象。
下面让我们来看一个对应的例子,类的内容如下:
import java.io.*; import java.util.Date; /** * 对象的序列化和反序列化测试类. * @author <a href="mailto:xiexingxing1121@126.com">AmigoXie</a> * @version 1.0 * Creation date: 2007-9-15 - 下午21:45:48 */ public class ObjectSaver { /** * @param args * @author <a href="mailto:xiexingxing1121@126.com">AmigoXie</a> * Creation date: 2007-9-15 - 下午21:45:37 */ public static void main(String[] args) throws Exception { ObjectOutputStream out = new ObjectOutputStream (new FileOutputStream("D:""objectFile.obj")); //序列化对象 Customer customer = new Customer("阿蜜果", 24); out.writeObject("你好!"); out.writeObject(new Date()); out.writeObject(customer); out.writeInt(123); //写入基本类型数据 out.close(); //反序列化对象 ObjectInputStream in = new ObjectInputStream (new FileInputStream("D:""objectFile.obj")); System.out.println("obj1=" + (String) in.readObject()); System.out.println("obj2=" + (Date) in.readObject()); Customer obj3 = (Customer) in.readObject(); System.out.println("obj3=" + obj3); int obj4 = in.readInt(); System.out.println("obj4=" + obj4); in.close(); } } class Customer implements Serializable { private String name; private int age; public Customer(String name, int age) { this.name = name; this.age = age; } public String toString() { return "name=" + name + ", age=" + age; } } |
输出结果如下:
obj1=你好! obj2=Sat Sep 15 22:02:21 CST 2007 obj3=name=阿蜜果, age=24 obj4=123 |
因此例比较简单,在此不再详述。
二.实现Serializable接口
ObjectOutputStream只能对Serializable接口的类的对象进行序列化。默认情况下,ObjectOutputStream按
照默认方式序列化,这种序列化方式仅仅对对象的非transient的实例变量进行序列化,而不会序列化对象的transient的实例变量,也不会序列
化静态变量。
当ObjectOutputStream按照默认方式反序列化时,具有如下特点:
1) 如果在内存中对象所属的类还没有被加载,那么会先加载并初始化这个类。如果在classpath中不存在相应的类文件,那么会抛出ClassNotFoundException;
2) 在反序列化时不会调用类的任何构造方法。
如果用户希望控制类的序列化方式,可以在可序列化类中提供以下形式的writeObject()和readObject()方法。
private void writeObject(java.io.ObjectOutputStream out) throws IOException private void readObject(java.io.ObjectInputStream in) throws IOException, ClassNotFoundException; |
当ObjectOutputStream对一个Customer对象进行序列化时,如果该对象具有writeObject()方法,那么就会执行这一方
法,否则就按默认方式序列化。在该对象的writeObjectt()方法中,可以先调用ObjectOutputStream的
defaultWriteObject()方法,使得对象输出流先执行默认的序列化操作。同理可得出反序列化的情况,不过这次是
defaultReadObject()方法。
有些对象中包含一些敏感信息,这些信息不宜对外公开。如果按照默认方式对它们序列化,那么它们的序列化数据在网络上传输时,可能会被不法份子窃取。对于这类信息,可以对它们进行加密后再序列化,在反序列化时则需要解密,再恢复为原来的信息。
默认的序列化方式会序列化整个对象图,这需要递归遍历对象图。如果对象图很复杂,递归遍历操作需要消耗很多的空间和时间,它的内部数据结构为双向列表。
在应用时,如果对某些成员变量都改为transient类型,将节省空间和时间,提高序列化的性能。
三. 实现Externalizable接口
Externalizable接口继承自Serializable接口,如果一个类实现了Externalizable接口,那么将完全由这个类控制自身的序列化行为。Externalizable接口声明了两个方法:
public void writeExternal(ObjectOutput out) throws IOException public void readExternal(ObjectInput in) throws IOException , ClassNotFoundException |
前者负责序列化操作,后者负责反序列化操作。
在对实现了Externalizable接口的类的对象进行反序列化时,会先调用类的不带参数的构造方法,这是有别于默认反序列方式的。如果把类的不带
参数的构造方法删除,或者把该构造方法的访问权限设置为private、默认或protected级别,会抛出
java.io.InvalidException: no valid constructor异常。
四. 可序列化类的不同版本的序列化兼容性
凡是实现Serializable接口的类都有一个表示序列化版本标识符的静态变量:
private static final long serialVersionUID; |
以上serialVersionUID的取值是Java运行时环境根据类的内部细节自动生成的。如果对类的源代码作了修改,再重新编译,新生成的类文件的serialVersionUID的取值有可能也会发生变化。
类的serialVersionUID的默认值完全依赖于Java编译器的实现,对于同一个类,用不同的Java编译器编译,有可能会导致不同的
serialVersionUID,也有可能相同。为了提高哦啊serialVersionUID的独立性和确定性,强烈建议在一个可序列化类中显示的定
义serialVersionUID,为它赋予明确的值。显式地定义serialVersionUID有两种用途:
1) 在某些场合,希望类的不同版本对序列化兼容,因此需要确保类的不同版本具有相同的serialVersionUID;
2) 在某些场合,不希望类的不同版本对序列化兼容,因此需要确保类的不同版本具有不同的serialVersionUID。
发表评论
-
Sington类的序列化留存
2011-08-01 14:30 1337Sington类的序列化 为了使Singleton类变成 ... -
两个数字互换位置的最简单办法
2009-03-11 17:50 1656a=1 ,b=2 a=a^b; b=a^b; a=a^b ... -
JAVA中各类CACHE机制实现的比较
2008-11-21 17:02 1673这几天学习ruby on rails,发现rails中的cac ... -
JAVA Clone机制
2008-11-10 16:24 47561,什么是Clone ? 简单地说, Clo ... -
实现对一个二维数组按指定的列集进行排序
2008-11-07 14:24 4886# private int array[][] = new ... -
各种排序算法java实现
2008-10-30 17:32 883原文地址 http://blog.csdn.net/lscho ... -
ini文件的读取
2008-08-14 17:54 1033import java.io.File; import jav ... -
原码,补码,反码
2008-02-26 16:03 1254原码按位取反成为反码; 反码+1成为补码 ... -
Java中的volatile关键字
2008-02-01 18:06 8060关于volatile ... -
Java语言中关键字讲解(一)strictfp的用途
2008-01-31 18:03 1558Java语言中关键字strictfp的用途- - 自Jav ... -
向上转型与接口回调的区别
2008-01-31 10:56 45011 .接口回调是什么 ... -
JSP页面中的pageEncoding和contentType
2008-01-30 15:56 1135关于JSP页面中的pageEncoding和contentTy ... -
Tomcat的配置技巧精华详解
2008-01-30 15:23 9421、配置系统管理(Admin ... -
Tomcat简介
2008-01-28 18:27 30171.Tomcat简介 Jakarta Tomcat服务 ... -
properties文件的Eclipse插件
2008-01-21 12:23 1025在此想和大家分享一个不错的编写properties文件的Ecl ... -
java---hashtable
2008-01-17 16:48 1959Hashtable检索数据时,必须比较所用关键字是否 ... -
Java泛型编程指南
2008-01-16 15:07 1573Java泛型编程指南 此系列文章译自SUN的泛型编程指南, ... -
Servlet/JSP学习笔记-Session和作用域
2008-01-16 14:13 2948Servlet/JSP学习笔记(7)-Session和作用域 ...
相关推荐
本教程"【IT十八掌徐培成】Java基础第02天-01.java关键字"深入讲解了Java的关键字及其用途。 Java的关键字主要分为以下几类: 1. 数据类型关键字:`byte`, `short`, `int`, `long`, `float`, `double`, `char`, `...
关键字是Java语言预定义的具有特殊含义的词汇,它们在程序中有着特定的作用。下面将详细讲解Java的关键字以及命名规范。 1. 关键字: - `abstract`:用于声明抽象类或抽象方法,抽象类不能实例化,抽象方法没有...
这个PPT教案详细地讲解了Java语言的基础语法,包括标识符的规则、关键字的作用以及常量和变量的使用,这些都是初学者掌握Java编程的基石。通过深入理解和实践这些基本概念,程序员能够编写出符合标准且易于理解的...
本章节主要讲解Java的序列化机制,包括Serializable接口、Externalizable接口、transient关键字、serialVersionUID变量等。 Java学习系列(十三):Java面向对象之界面编程 本章节主要讲解Java的界面编程,包括AWT...
### Java关键字及其用法 Java作为一种广泛使用的编程语言,拥有丰富的关键字集,这些关键字是预定义的保留字,具有特定的语法含义。下面将详细解释部分重要的Java关键字及其使用场景。 #### Abstract `abstract`...
- 不能是Java关键字。 - 区分大小写。 - **命名约定**: - 包名:所有字母都小写。 - 类名/接口名:每个单词首字母大写。 - 变量名/方法名:首单词首字母小写,后续单词首字母大写。 - 常量名:所有字母大写,...
关于Java关键字,如transient、instanceof、final、static等,要求应聘者不仅要了解其含义,还要掌握其用法。 集合类的知识点包括了解不同集合类的特点和使用场景,例如ArrayList、LinkedList、Vector的区别,...
4. **枚举和注解**:介绍了枚举类型在Java中的强大功能,如枚举常量、枚举方法、枚举开关语句等,并讲解了自定义注解的创建和使用,以及元注解的应用。 5. **方法**:提倡使用重载而非覆盖,解释了如何有效地使用...
本章节将围绕“Java基础语法”展开讲解,主要内容包括Java语言中的关键字、标识符、注释、常量和变量等核心概念。这部分是学习Java编程的基础,对于初学者来说至关重要。 #### 2.1 关键字 关键字是Java语言中具有...
《Effective Java》讲解了如何正确使用synchronized关键字,如何实现线程安全的类,以及线程池和并发工具类的使用。 7. **方法覆盖与重载**:Bloch解释了这两个概念的区别,以及如何在设计API时明智地使用它们,以...
在每个概念的讲解中,都会配有Java代码实现,以及运行结果,以便于理解和实践。 作者Brad Rubin是一位在Java安全性和教育领域有着深厚经验的专家,他在IBM和Imation Corp.的工作经历使他对于Java平台的安全性有深入...
5. **序列化**:Java的Serializable接口使得对象可以被序列化和反序列化,书中会介绍如何实现序列化,处理transient关键字,以及使用ObjectInputStream和ObjectOutputStream。 6. **JVM内部**:了解JVM的工作原理对...
JAVA学习笔记.pdf 中讲解了JAVA语言的基础知识,包括类的基本知识、成员变量、成员方法、类的实例、内部类、匿名类、接口、包等。 类的基本知识 在JAVA中,类是对象的蓝图,类的声明语法为:[访问控制符] class ...
《Java提高篇(二一)-----ArrayList.pdf》这篇文章主要深入讲解了Java编程语言中的ArrayList类的内部工作机制以及如何高效地使用这个重要的数据结构。ArrayList是基于数组实现的动态数组,用于存储顺序的集合,并且...
文档内容分为多个章节,从Java的基本语法开始,详细介绍了关键字的使用,如static、final、transient、volatile、synchronized、import、enum等,这些关键字在Java程序中有着非常重要的作用。例如,static关键字可以...
下面将详细讲解Java语言及其他相关的知识点。 1. Java 数据类型的取值范围: - `short` 型的取值范围是 `-2^15`(即-32768)到 `2^15 - 1`(即32767)。在给定的选项中,正确答案是C. `-2^15--- 2^15-1`。 2. ...
这份PPT课件涵盖了Java语言的关键概念,主要包括字符集、保留字、基本数据类型、变量与常量、运算符与表达式以及流程控制语句,并通过实例进行讲解。 首先,Java的字符集基于Unicode,它是一种16位的编码系统,能够...
面向对象是Java编程的核心概念,本章主要讲解了关于Java中的类与对象、成员变量与成员函数、对象的初始化与构造函数、类的封装性、继承性、多态性、抽象类与抽象函数,以及类的静态变量与静态函数。 首先,类是Java...