java--流相关知识点

摘自《疯狂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序列化算法机制

  1. 所有保存到磁盘中的对象都有一个序列化编号
  2. 当程序试图序列化一个对象时,程序将先检查该对象是否已经被序列化过,只有该对象从未(在本次虚拟机中)被序列化过,系统才会将该对象转换成字节序列并输出
  3. 如果某个对象已经序列化过,程序将直接输出一个序列号,而不是再次从新序列化对象

由于java序列化机制,如果多次系列化同一个java’对象,只有第一次序列化时才会把该java对象转化成字节序列并输出,,潜在问题是当程序序列化一个可变对象时,只有第一次使用writeObject()方法输出时才会将该对象转化成字节序列,当程序再次调用writeObject()方法时,程序只是输出当前的序列化编号,即使后面该对象的实例变量值已经被改变,改变的实例变量值也不会输出

在反序列化是确保序列化版本的兼容,最好在每个序列化的类中加入private static final long serialVersionUID这个变量

修改类对serialVersionUID的影响
1. 如果修改类时仅仅修改了方法,则反序列化不受任何影响,类定义无需修改serialVersionUID类的变量的值
2. 如果修改类时仅仅修改了静态变量或瞬态实例变量,则反序列化不受任何影响,类定义无须修改serialVersionUID
3. 如果修改类时修改了非瞬态变量的实例变量,则可能导致序列化版本不兼容。如果对象流中的对象和新类中包含同名的实例变量,而实例变量类型不同,则反序列化失败,类定义应该更新serialVersionUID。如果对象流中的对象比新类中包含更多的实例变量,则多出的实例变量值被忽略,序列化版本可以兼容,类定义可以不更新serialVersionUID,但是反序列化得到的新对象中多出的实例变量值都是null(引用类型实例变量)或0(基本类型实例变量)

关于序列化的注意

  1. 对象的类名,实例变量(包括基本类型、数组、对其他对象引用)都会被序列化:方法、类变量(即static修饰的成员变量)、transient实例变量都不会被序列化
  2. 实现Serializable接口的类如果需要让某个实例变量不被序列化,则可以在该实例变量前加transient修饰符,而不是static关键字,虽然static关键字也可以达到这个效果,但static关键字不能这样用。
  3. 保证序列化对象的实例变量类型也是可序列化的,否则需要使用transient修饰该变量
  4. 反序列化时对象时必须有序列化对象的class文件
  5. 当通过文件、网络来读取序列化后的对象时,必须按照实际写入的顺序读取
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值