- 浏览: 369233 次
- 性别:
- 来自: 成都
文章分类
最新评论
-
tuspark:
关于javadoc这里讲解的更全面:《javadoc设置》。
Eclipse中生成javadoc【Z】 -
yuexiang1007:
帮我解决了问题,谢谢!!!
java.math.BigInteger使用心得总结 -
netwelfare:
个人感觉,文章对HashMap的遍历分析的有点浅,不如这里的介 ...
HashMap遍历的两种方式【Z】 -
memoryisking:
关于java.math.BigInteger讲解在这里可以看到 ...
java.math.BigInteger使用心得总结 -
巴尾的兔兔帅:
divide应该是除吧?不是减。dividepublic Bi ...
java.math.BigInteger使用心得总结
继承对序列化的影响
这篇文章最初的起源是来自于我在工作中碰到一个问题。
我的一个聪明的同事自己写了一个框架,这个框架用来处理
Client与Server之间的通讯,在其中不免用到一些继承,从
我这个角度上看,他的继承用的过于复杂,这样产生了许多
副作用,其中一个就是和对象序列化相关。好了,闲话少说
,开始进入正文。
当一个父类实现Serializable接口后,他的子类都将自动的实现序列化。
一、序列化简介
在这里我先简要介绍一下和序列化相关的一些概念,如
果你觉得自己对序列化已经比较熟悉,那么跳过这一节,直
接看下一节内容。
序列化是Java中的一个非常重要的特性,通过序列化机
制,我们可以将Java的对象变成流,或者存储在硬盘中,或
者通过网络传输给网络的其他用户。而序列化在RMI,EJB中
都有应用,可以说是构成J2EE的一个重要的技术。
1)Serializable接口
如果想让一个类可被序列化,那么这个类必须实现
Serializable接口,这个接口本身没有任何方法和属性,它
的作用只是为了标示一个类可以被序列化,这一个特性好像
在Java里用的比较多,比如克隆也是采用了相同的机制。
因此,如果我们想创建一个可以被序列化的类,我们可
以像下面的代码一样做。
import java.io.Serializable;
public class SerialClass implements Serializable
{
private static final long serialVersionUID = -
190734710746841476L;
private String c;
public int a;
public String b;
public int getA(){return a;}
public String getB(){return b;}
public void setA(int a){this.a = a;}
public void setB(String b){this.b = b;}
public String getC(){return c; }
public void setC(String c){this.c = c; }
从上面的例子我们可以看到,除了需要实现
Serialzable接口外,一个可序列化的类和一个普通类没有
什么大的区别。不过我们还是有一些特别的东西需要注意的
。
2)属性 serialVersionUID
这个属性是一个私有的静态final属性,一般刚接触序
列化的人会忽略这个属性,因为这个属性不是必须的。这个
属性主要是为了保证一个可序列化类在不同的Java编译器中
,能保证相同的版本号,这样保证当这个类在不同的JVM中
进行序列化与反序列化时不会出现InvalidClassException
异常。
3)属性与序列化
那么我们再考虑另一个问题,是不是一个类实现了
Serializable之后就可以看成是可序列化的类呢?答案是不
行,实际上一个类是否可序列化还需要看这个类的属性,在
Java规范中,一个类是否能序列化,不但取决于这个类是否
实现Serializable接口,还取决于这个类中的属性是否可序
列化。在上面的例子里,一共有三个属性,两个是String对
象,一个是int类型的属性,实际上,String类是可序列化
的(继承了Serializable 接口且它的属性都是可序列化的
),而像int这样的基本数据类型在Java中被看成是可序列
化的,因此我们可以确认上面的这个类是一个可序列化的类
。那么现在我们已经创建了一个可序列化类,接下去的问题
是怎么使用这个类,对它进行序列化与反序列化操作?
4)ObjectOutputStream 和 ObjectInputStream类
Jdk提供了两个IO流类进行序列化和反序列化操作,其
中ObjectOutputStream中的writeObject方法对一个对象进
行序列化工作;而ObjectInputStrem中的readObject方法则
负责发序列化的操作。下面的例子完整的显示了一个类的序
列化和反序列化操作。
package serialtest;
import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.ObjectInputStream;
import java.io.ObjectOutputStream;
public class TestClass
{
public static void main(String[] args) throws
Exception
{
SerialClass s = new SerialClass();
s.setA(10);
s.setB("hello");
s.setC("world");
//创建ObjectOutputStream对象,准备序列化对象s
ObjectOutputStream oos = new
ObjectOutputStream(
new FileOutputStream("abc"));
//调用writeObject将对象进行序列化,存储在文件
abc中。
oos.writeObject(s);
oos.flush();
oos.close();
//创建ObjectInputStream对象,准备从文件abc中
反序列化SerialClass对象
ObjectInputStream ois = new
ObjectInputStream(
new FileInputStream("abc"));
//调用readObject将存储在文件abc中的字节流反序
列化为SerialClass对象。
s = (SerialClass) ois.readObject();
System.out.println(s.getA());
System.out.println(s.getB());
System.out.println(s.getC());
}
}
执行程序,最后的结果应该是:
10
Hello
World
好了,序列化的基础知识就讲到这里。接下来我们看看
继承对于序列化的影响。
二、继承对序列化的影响
我们现在把上面的例子做点变换,变成如下的形式。
1)首先创建一个Interface。
package serialtest;
public interface SuperInterface
{
public int getA();
public String getB();
public void setA(int a);
public void setB(String b);
}
2)创建一个抽象类实现SuperInterface接口,注意,这个
类没有实现Serializable接口。
package serialtest;
public abstract class AbsSuperClass implements
SuperInterface
{
public int a;
public String b;
public int getA() {return a;}
public String getB(){return b; }
public void setA(int a) {this.a = a;}
public void setB(String b) {this.b = b;}
}
3)SerialClass类继承了AbSuperClass,同时它也实现了
Serializable接口。
package serialtest;
import java.io.Serializable;
public class SerialClass extends AbsSuperClass
implements Serializable
{
private static final long serialVersionUID = -
190734710746841476L;
private String c;
public String getC(){return c; }
public void setC(String c) {this.c = c;}
}
这时候我们在运行Test,将会发现结果和第一节的例子不一
样了,这时候结果将变成:
0
null
world
而导致这个结果的原因就在AbSuperClass上,因为
AbSuperClass没有实现Serializable接口,所以它不可被序
列化;但SerialClass由于实现了Serializable接口而成为
一个可序列化的类,这时候,属性c因为是SerialClass的类
所以c的值在序列化时将被保存,而属性a和b是从
AbSuperClass中继承而来,他们因为AbSuperClass的缘故而
导致了它们的值不能在序列化是被保存。
关于这一点,我在Java文档中也得到了证实,在Java
doc中明确指出,如果一个类继承了一个非Serializable类
时,如果想在序列化中保存父类的属性,则需要实现额外的
代码显式地存储父类的属性值。
最后再回到本文的初衷,诚然,这篇文章是以对象的序
列化为主的,但最后我又要回到一个比较古老的话题上,那
就是在软件开发的时候,继承到底该用到什么一种程度?毕
竟,滥用继承对整个系统的影响将是深远的。
[参考文献]
1. Effective Java. Joshua Bloch著,潘爱民译。
2. Java doc. doc.sun.com.
3. Java 编程语言。 Ken Arnold等著
发表评论
-
JSTL中的<c:标签【Z】
2011-08-31 20:48 1146Taglib 伪指令 Java代码 ... -
java接口嵌套【Z】
2011-05-18 17:09 1342在Java语言中,接口可以嵌套在类或其它接口中。由于Jav ... -
JMF安装【Z】
2011-05-07 20:52 1094下载并安装JMF 在MyEclipse中选择 窗口/首选项/ ... -
javaMail【Z】
2011-04-18 16:12 1078commons-email-1.1.jar: 这是Apache ... -
java生成PDF文件【Z】
2011-01-14 15:03 1211如果应用程序需要动态 ... -
字符集编码【Z】
2010-12-29 18:54 893问题研究 --字符集 ... -
java常用第三方jar包【Z】
2010-12-26 22:39 3360commons-digester.jar Digeste ... -
不使用JNI获得系统信息【Z】
2010-12-22 13:28 589在Java中,可以获得总的物理内存、剩余的物理内存、已使用的物 ... -
java获得当前路径【Z】
2010-12-18 14:37 9421、利用System.getProperty()函数获取当前路 ... -
java容器类-1【Z】
2010-12-17 21:43 1013对象的集合 如果程序的对象数量有限 ... -
java UIManager的风格【Z】
2010-12-17 12:28 2671Java'中的几种Look and Feel 1、Met ... -
serialVersionUID的作用和意义【Z】
2010-12-15 10:38 1553在Java中,软件的兼容性是一个大问题,尤其在使用到对象串行性 ... -
HashMap遍历的两种方式【Z】
2010-12-07 18:42 968HashMap遍历的两种方式 ... -
23个经典JDK设计模式——系统JDK使用设计模式的例子【Z】
2010-12-05 21:33 1029酷壳 版主陈皓近日发表博文《JDK里的设计模式 》,文中他 ... -
String/StringBuffer/StringBuild【Z】
2010-11-12 14:10 1346[编辑] String,StringBuffer和String ... -
tomcat中web.xml详解【Z】
2010-10-18 14:47 9251 定义头和根元素 部 ... -
Java 理论与实践: 线程池与工作队列【Z】
2010-10-18 09:45 932线程池有助于实现最 ... -
单例模式完全剖析【Z】
2010-10-17 22:48 790Buildfile: build.xml ... -
在MyEclipse中直接查看class文件(在没有源码的情况下)【Z】
2010-10-17 22:04 1309想直接在myeclipse中查看class文件,就像查看普通的 ... -
HashMap实现及冲突【Z】
2010-10-12 21:41 962了解 HashMap 原理对于日后的缓存机制多少有 ...
相关推荐
- **1_5 Java Networking**: 这部分可能介绍Java如何进行网络通信,包括Socket编程、URL类、HTTP协议的实现,以及数据的序列化和反序列化。 6. **Java平台与J2EE** - **3_1 J2EEOverview** 和 **3_2 Java EE 5 ...
6. **文件操作**:为了保存和加载游戏进度,可能需要使用到文件操作,如序列化对象到文件,或者读取文件恢复游戏状态。 7. **单元测试**:为了确保代码的正确性,开发者可能使用JUnit等测试框架对游戏逻辑进行单元...
7. **第11周**:十一周可能介绍I/O流,包括文件读写、字符流、字节流、缓冲流以及对象序列化,这些都是处理数据传输的关键技术。 8. **第12周**:十二周可能涉及Java的反射API,这是一个强大的工具,允许程序在运行...
11. **存档与加载**:为了让玩家可以保存进度,开发者需要实现游戏进度的序列化和反序列化功能,可以使用Java的内置序列化或自定义格式。 12. **测试与调试**:游戏开发过程中,单元测试、集成测试和调试是必不可少...
文件中可能还涉及到了Java I/O流,包括文件读写、网络通信、对象序列化等,这些都是Java进行数据传输和持久化存储的重要手段。同时,线程和并发处理也是Java编程中的重要部分,文件可能会包含线程的创建、同步、中断...
5. **输入/输出流**:学习文件操作、序列化、缓冲区和对象流等IO概念。 6. **多线程**:线程的创建、同步机制(synchronized关键字、wait()、notify()、notifyAll()方法)、线程池和并发工具类。 7. **网络编程**...
6. **序列化与反序列化**:Java的序列化机制允许对象的状态被保存到磁盘或在网络上传输。这可能用于在聊天过程中暂时保存用户的状态,或者在网络通信中将对象转换为可传输的字节流。 7. **事件监听器**:Swing组件...
12. **输入输出流**:Java提供了丰富的I/O流API处理数据的输入和输出,包括文件流、网络流、对象序列化等。 13. **集合框架**:Java集合框架包括List、Set、Queue等接口,以及ArrayList、LinkedList、HashSet、...
4. **输入/输出流**:Java的I/O流系统是处理数据输入和输出的关键,笔记会涵盖文件操作、字节流和字符流、缓冲流、对象序列化等主题。 5. **多线程**:Java内置了强大的多线程支持,笔记会讲述如何创建和管理线程,...
6. **IO流**:字节流与字符流的区别、缓冲流的使用、转换流、对象序列化、File类的操作、输入输出流的关闭策略、NIO(New IO)。 7. **数据库操作**:SQL语句的编写与优化(JOIN、索引、子查询)、事务处理(ACID...
开发者可以借此学习到如何处理网络连接、数据包的序列化和反序列化、以及错误处理等技巧。 再者,游戏的图形界面是用户体验的重要组成部分。Java提供了JavaFX和Swing等图形用户界面库,可以创建丰富的2D和3D图形。...
Java 期末考试题库,包含基础题单选题 18 道,涵盖 Java 语言基础知识点,包括变量、数据类型、运算符、控制流程、数组、继承关系、方法和类的定义等。 一、单选题: 1. 下列语句序列执行后,k 的值是(B)。这是...
5. **输入/输出流**:介绍了Java I/O流的概念,包括文件读写、序列化、网络I/O,以及NIO(非阻塞I/O)框架,这对于数据传输和持久化存储至关重要。 6. **异常处理**:学习如何正确地捕获和处理程序运行时可能出现的...
5. **输入/输出流**:介绍Java I/O系统,包括文件操作、字节流和字符流,以及缓冲区和对象序列化。 其次,"实例程序"标签表明本书注重实践,书中可能包含: 6. **实例分析与编写**:每个概念都会伴随有实际的代码...
文件系统操作和序列化技术在这里起到关键作用。 网络功能在多人在线游戏中尤为重要。Java的Socket编程允许开发者创建服务器和客户端,实现实时的多人对战。通过网络通信,玩家间的信息交换(如位置、动作)可以实时...
4. **IO流**:Java的IO流库用于读写数据,包括字符流和字节流,以及高级的缓冲流、对象序列化和文件操作。 5. **多线程**:Java内置了对多线程的支持,允许程序同时执行多个任务,通过Thread类和Runnable接口实现。...
7. **对象序列化**:为了在网络间传输复杂的对象,如坦克或游戏地图,开发者可能需要使用Java的序列化机制。这允许将对象转换为字节流,然后在网络上传输,到达目的地后再还原。 8. **游戏循环**:游戏通常有一个主...
- 使用 `DataInputStream` 和 `DataOutputStream` 可以更高效地处理结构化数据,例如序列化的对象。 5. **文件复制实例** - 在Java中,文件复制通常涉及使用 `BufferedInputStream` 和 `BufferedOutputStream` 的...
6. **输入输出流**:Java的IO流系统是处理数据输入和输出的核心,包括字节流、字符流、缓冲流、对象序列化等,对于文件操作和网络通信尤其重要。 7. **网络编程**:Java提供了Socket编程接口,允许开发网络应用程序...
- `transient`:用于标记对象的状态不被序列化。 - `try`:用于捕获异常。 - `volatile`:用于标记共享变量以保证可见性。 ### 6. Java 数据类型 Java 支持两种类型的数据类型:基本数据类型和引用数据类型。 - *...