摘自《疯狂Java讲义》
关闭流
在使用处理流包装了底层节点流之后,关闭输入/输出资源时,只要关闭最上层的处理流即可。关闭最上层的处理流时,系统会自动关闭被该处理流包装的节点流
缓冲流
增加缓冲流功能可以提高输入输出的效率,增加缓冲功能后需要使用flush方法才可以将缓冲区的内容写入实际的物理节点
RandomAccessFile向文件追加内容
public class Test {
public static void main(String[] args) throws Exception {
RandomAccessFile raf = new RandomAccessFile("out.txt","rw");
try{
raf.seek(raf.length());
raf.write("追加内容".getBytes());
}catch (Exception e){
e.printStackTrace();
}
finally {
raf.close();
}
}
}
RandomAccessFile依然不能向文件的指定位置插入内容,如果直接将文件记录指针移动到中间某位置后开始输出,则新输出的内容会覆盖文件中原有的内容,如果需要向指定位置输入内容,程序需要先把插入点后面的内容读入缓冲区,等把需要插入的数据写入文件后,在将缓冲区的内容追加到文件后面
public class Test {
public static void main(String[] args) throws Exception {
insert("Test",45,"插入的内容");
}
public static void insert(String fileName,long pos,String insertContent)throws IOException{
File temp = File.createTempFile("tmp",null);
temp.deleteOnExit();
RandomAccessFile raf = new RandomAccessFile(fileName,"rw");
FileOutputStream tmpOut = new FileOutputStream(temp);
FileInputStream tmpIn = new FileInputStream(temp);
raf.seek(pos);
byte[] bbuf = new byte[64];
int hasRead = 0;
while ((hasRead = raf.read(bbuf)) > 0){
tmpOut.write(bbuf,0,hasRead);
}
raf.seek(pos);
raf.write(insertContent.getBytes());
while ((hasRead = tmpIn.read(bbuf)) > 0){
raf.write(bbuf,0,hasRead);
}
}
}
反序列化
如果使用序列化机制向文件中写入了多个Java对象,使用反序列化机制恢复对象时必须按实际写入的顺序读取。
当一个可序列化类有多个父类时(包括直接父类和间接父类),这些父类要么有无参数的构造器,要么也是可序列化的,否则反序列化将抛出InvalidClassException异常,如果父类是不可序列化的,只是带有无参数的构造器,则该父类定义的成员变量值不会序列化到二进制流中。
java序列化算法机制
- 所有保存到磁盘中的对象都有一个序列化编号
- 当程序试图序列化一个对象时,程序将先检查该对象是否已经被序列化过,只有该对象从未(在本次虚拟机中)被序列化过,系统才会将该对象转换成字节序列并输出
- 如果某个对象已经序列化过,程序将直接输出一个序列号,而不是再次从新序列化对象
由于java序列化机制,如果多次系列化同一个java’对象,只有第一次序列化时才会把该java对象转化成字节序列并输出,,潜在问题是当程序序列化一个可变对象时,只有第一次使用writeObject()方法输出时才会将该对象转化成字节序列,当程序再次调用writeObject()方法时,程序只是输出当前的序列化编号,即使后面该对象的实例变量值已经被改变,改变的实例变量值也不会输出
在反序列化是确保序列化版本的兼容,最好在每个序列化的类中加入private static final long serialVersionUID
这个变量
修改类对serialVersionUID的影响
1. 如果修改类时仅仅修改了方法,则反序列化不受任何影响,类定义无需修改serialVersionUID类的变量的值
2. 如果修改类时仅仅修改了静态变量或瞬态实例变量,则反序列化不受任何影响,类定义无须修改serialVersionUID
3. 如果修改类时修改了非瞬态变量的实例变量,则可能导致序列化版本不兼容。如果对象流中的对象和新类中包含同名的实例变量,而实例变量类型不同,则反序列化失败,类定义应该更新serialVersionUID。如果对象流中的对象比新类中包含更多的实例变量,则多出的实例变量值被忽略,序列化版本可以兼容,类定义可以不更新serialVersionUID,但是反序列化得到的新对象中多出的实例变量值都是null(引用类型实例变量)或0(基本类型实例变量)
关于序列化的注意
- 对象的类名,实例变量(包括基本类型、数组、对其他对象引用)都会被序列化:方法、类变量(即static修饰的成员变量)、transient实例变量都不会被序列化
- 实现Serializable接口的类如果需要让某个实例变量不被序列化,则可以在该实例变量前加transient修饰符,而不是static关键字,虽然static关键字也可以达到这个效果,但static关键字不能这样用。
- 保证序列化对象的实例变量类型也是可序列化的,否则需要使用transient修饰该变量
- 反序列化时对象时必须有序列化对象的class文件
- 当通过文件、网络来读取序列化后的对象时,必须按照实际写入的顺序读取