zl程序教程

您现在的位置是:首页 >  后端

当前栏目

Java 基础 (IO 对象流,随机存储文件流)

2023-09-27 14:26:01 时间

IO 对象流

ObjectInputStream 和 OjbectOutputSteam

用于存储和读取基本数据类型数据或对象的处理流。它的强大之处就是可以把Java中的对象写入到数据源中,也能把对象从数据源中还原回来。

  • 序列化: 用 ObjectOutputStream 类保存基本类型数据或对象的机制
  • 反序列化: 用 ObjectInputStream 类读取基本类型数据或对象的机制
  • ObjectOutputStream 利 ObjectInputStream 不能序列化 static 和transient 修饰的成员变量

对象序列化机制允许把内存中的 Java 对象转换成平台无关的二进制流,从而允许把这种二进制流持久地保存在磁盘上,或通过网络将这种二进制流传输到另一个网络节点。当其它程序获取了这种二进制流,就可以恢复成原来的 Java 对象

序列化的好处在于可将任何实现了 Serializable 接口的对象转化为字节数据,使其在保存和传输时可被还原

如果需要让某个对象支持序列化机制,则必须让对象所属的类及其属性是可序列化的,为了让某个类是可序列化的,该类必须实现如下两个接口之一。否则,会抛出 NotSerializableException 异常

> Serializable
> Externalizable

凡是实现 Serializable 接口的类都有一个表示序列化版本标识符的静态变量;

> private static final long serialVersionUID;
> serialVersionUID 用来表明类的不同版本间的兼容性。简言之,其目的是以序列化对象进行版本控制,有关各版本反序列化时是否兼容。
> 如果类没有显示定义这个静态常量,它的值是Java运行时环境根据类的内部细节自动生成的。若类的实例变量做了修改,seriaVersionUID 可能发生变化。故建议,显式声明。

Person.java

package com.klvchen.java;

import java.io.Serializable;

public class Person implements Serializable {

    public static final long serialVersionUID = 475684125L;

    private String name;
    private int age;

    public Person(String name, int age) {
        this.name = name;
        this.age = age;
    }

    public Person() {
    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public int getAge() {
        return age;
    }

    public void setAge(int age) {
        this.age = age;
    }

    @Override
    public String toString() {
        return "Person{" +
                "name='" + name + '\'' +
                ", age=" + age +
                '}';
    }
}

ObjectInputOutputStreamTest.java

package com.klvchen.java;

import org.junit.Test;

import java.io.*;

public class ObjectInputOutputStreamTest {
    /*
    序列化过程:将内存中的 java 对象保存到磁盘中或通过网络传输出去
    使用 ObjectOutputStream 实现
     */

    @Test
    public void testObjectOutputStream(){
        ObjectOutputStream oos = null;

        try {
            oos = new ObjectOutputStream(new FileOutputStream("object.dat"));
            oos.writeObject(new String("我爱北京天安门"));
            oos.flush();

            oos.writeObject(new Person("wangyi", 23));
            oos.flush();

        } catch (IOException e) {
            e.printStackTrace();
        } finally {
            if (oos != null) {
                try {
                    oos.close();
                } catch (IOException e) {
                    e.printStackTrace();
                }
            }
        }

    }

    /*
    反序列化:将磁盘文件中的对象还原为内存中的一个 java 对象
    使用ObjectInputStream 来实现
     */
    @Test
    public void testObjectInputStream(){
        ObjectInputStream ois = null;

        try {
            ois = new ObjectInputStream(new FileInputStream("object.dat"));

            Object obj = ois.readObject();
            String str = (String) obj;

            Person p = (Person) ois.readObject();

            System.out.println(str);
            System.out.println(p);

        } catch (IOException | ClassNotFoundException e) {
            e.printStackTrace();
        } finally {
            if (ois != null) {
                try {
                    ois.close();
                } catch (IOException e) {
                    e.printStackTrace();
                }
            }
        }
    }
}

IO 随机存储文件流 RandomAccessFile

* RandomAccessFile 声明在 java.io 包下,但直接继承于java.lang.Object类。并且它实现了 DataInput、DataOutput 这两个接口,也就意味着这个类既可以读也可以写。

* RandomAccessFile 类支持“随机访问”的方式,程序可以直接跳到文件的任意地方来读、写文件
> 支持只访问文件的部分内容
> 可以向已存在的文件后追加内容

* RandomAccessFile 对象包含一个记录指针,用以标示当前读写处的位置。RandomAccessFile 类对象可以自由移动记录指针:
> long getFilePointer(): 获取文件记录指针的当前位置
> void seek(long pos): 将文件记录指针定位到 pos 位置

* 构造器
public RandomAccessFile(File file,String mode)
public RandomAccessFile(String name,String mode)

* 创建 RandomAccessFile 类实例需要指定一个 mode参数,该参数指定RandomAccessFile 的访问模式:
>r: 以只读方式打开
>rw: 打开以便读取和写入
>rwd: 打开以便读取和写入;同步文件内容的更新
>rws: 打开以便读取和写入;同步文件内容和元数据的更新

* 如果模式为只读r。则不会创建文件,而是会去读取一个已经存在的文件,如果读取的文件不存在则会出现异常。如果模式为rw读写。如果文件不存在则会去创建文件,如果存在则不会创建。

如果 RandomAccessFile 作为输出流时,写出到的文件如果不存在,则在执行过程中自动创建如果写出到的文件存在,则会对原有文件内容进行覆盖。(默认情况下,从头覆盖>|

hello.txt

abcdefghijklm

RandomAccessFileTest.java

package com.klvchen.java;

import org.junit.Test;

import java.io.File;
import java.io.FileNotFoundException;
import java.io.IOException;
import java.io.RandomAccessFile;
import java.nio.charset.StandardCharsets;

public class RandomAccessFileTest {

    @Test
    public void test1()  {
        RandomAccessFile raf1 = null;
        RandomAccessFile raf2 = null;
        try {
            raf1 = new RandomAccessFile(new File("1.png" ),"r");
            raf2 = new RandomAccessFile(new File("2.png"), "rw");

            byte[] buffer = new byte[1024];
            int len;
            while ((len = raf1.read(buffer)) != -1) {
                raf2.write(buffer, 0 ,len);
            }
        } catch (IOException e) {
            e.printStackTrace();
        } finally {
            if (raf1 != null) {
                try {
                    raf1.close();
                } catch (IOException e) {
                    e.printStackTrace();
                }
            }

            if (raf2 != null) {
                try {
                    raf2.close();
                } catch (IOException e) {
                    e.printStackTrace();
                }
            }
        }
    }

    @Test
    public void test2() throws IOException {
        RandomAccessFile raf1 = new RandomAccessFile("hello.txt", "rw");

        raf1.seek(3); //将指针调到角标为3的位置
        //保存指针3后面的所有数据到StringBuilder中
        StringBuilder builder = new StringBuilder((int) new File("hello.txt").length());

        byte[] buffer = new byte[20];
        int len;
        while ((len = raf1.read(buffer)) != -1) {
            builder.append(new String(buffer, 0, len));
        }

        //调回指针,写入"xyz"
        raf1.seek(3);
        raf1.write("xyz".getBytes());

        //将StringBuilder中的数据写入到文件中
        raf1.write(builder.toString().getBytes());

        raf1.close();
    }
}