前面已经把java io的主要操作讲完了

这一节我们来说说关于java io的其他内容

 

Serializable序列化

实例1:对象的序列化

​​import​​          ​​java.io.File;​​        


​​import​​ ​​java.io.FileOutputStream;​​


​​import​​ ​​java.io.IOException;​​


​​import​​ ​​java.io.ObjectOutputStream;​​


​​import​​ ​​java.io.Serializable;​​





​​@SuppressWarnings​​ ​​(​​ ​​"serial"​​ ​​)​​


​​//一个类要想实现序列化则必须实现Serializable接口​​


​​class​​ ​​Person ​​ ​​implements​​ ​​Serializable {​​


​​private​​ ​​String name;​​


​​private​​ ​​int​​ ​​age;​​





​​public​​ ​​Person(String name, ​​ ​​int​​ ​​age) {​​


​​this​​ ​​.name = name;​​


​​this​​ ​​.age = age;​​


​​}​​





​​public​​ ​​String toString() {​​


​​return​​ ​​"Name:"​​ ​​+ ​​ ​​this​​ ​​.name + ​​ ​​", Age:"​​ ​​+ ​​ ​​this​​ ​​.age;​​


​​}​​


​​}​​





​​public​​ ​​class​​ ​​Demo {​​


​​public​​ ​​static​​ ​​void​​ ​​main(String[] args) {​​


​​String path = File.separator + ​​ ​​"home"​​ ​​+ File.separator + ​​ ​​"siu"​​ ​​+​​


​​File.separator + ​​ ​​"work"​​ ​​+ File.separator + ​​ ​​"demo.txt"​​ ​​;​​





​​Person p1 = ​​ ​​new​​ ​​Person(​​ ​​"zhangsan"​​ ​​,​​ ​​12​​ ​​);​​


​​Person p2 = ​​ ​​new​​ ​​Person(​​ ​​"lisi"​​ ​​,​​ ​​14​​ ​​);​​





​​//此处创建文件写入流的引用是要给ObjectOutputStream的构造函数玩儿​​


​​FileOutputStream fos = ​​ ​​null​​ ​​;​​


​​ObjectOutputStream oos = ​​ ​​null​​ ​​;​​


​​try​​ ​​{​​


​​fos = ​​ ​​new​​ ​​FileOutputStream(path);​​


​​oos = ​​ ​​new​​ ​​ObjectOutputStream(fos);​​





​​//这里可以写入对象,也可以写入其他类型数据​​


​​oos.writeObject(p1);​​


​​oos.writeObject(p2);​​


​​} ​​ ​​catch​​ ​​(IOException e) {​​


​​e.printStackTrace();​​


​​} ​​ ​​finally​​ ​​{​​


​​try​​ ​​{​​


​​oos.close();​​


​​} ​​ ​​catch​​ ​​(IOException e) {​​


​​e.printStackTrace();​​


​​}​​


​​}​​


​​}​​


​​}​​


所谓对象序列化就是把一个对象进行持久化存储,方便保留其属性

通俗点说,等于把一个对象从堆内存里边揪出来放到硬盘上

当然,如果你开心,你可以序列化其他东西,包括数组,基本数据类型等等

来看看内容,神马玩意儿这是……


java中的IO操作总结(四)_java io 输入输出

 

实例2:对象的反序列化

​​import​​          ​​java.io.File;​​        


​​import​​ ​​java.io.FileInputStream;​​


​​import​​ ​​java.io.IOException;​​


​​import​​ ​​java.io.ObjectInputStream;​​





​​public​​ ​​class​​ ​​Demo {​​


​​public​​ ​​static​​ ​​void​​ ​​main(String[] args) {​​


​​String path = File.separator + ​​ ​​"home"​​ ​​+ File.separator + ​​ ​​"siu"​​ ​​+​​


​​File.separator + ​​ ​​"work"​​ ​​+ File.separator + ​​ ​​"demo.txt"​​ ​​;​​





​​//好吧,这里代码写得着实有点长了,还要抛异常什么的​​


​​//如果你也看的烦,那就在主方法上抛吧,构造方法里用匿名对象就好了​​


​​//什么?别告诉我你不知道匿名对象​​


​​FileInputStream fis = ​​ ​​null​​ ​​;​​


​​ObjectInputStream ois = ​​ ​​null​​ ​​;​​


​​try​​ ​​{​​


​​fis = ​​ ​​new​​ ​​FileInputStream(path);​​


​​ois = ​​ ​​new​​ ​​ObjectInputStream(fis);​​





​​//这里返回的其实是一个Object类对象​​


​​//因为我们已知它是个Person类对象​​


​​//所以,就地把它给向下转型了​​


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


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





​​//抛死你,烦烦烦~!!!​​


​​} ​​ ​​catch​​ ​​(IOException e) {​​


​​e.printStackTrace();​​


​​} ​​ ​​catch​​ ​​(ClassNotFoundException e) {​​


​​e.printStackTrace();​​


​​} ​​ ​​finally​​ ​​{​​


​​try​​ ​​{​​


​​//还是要记得关闭下流​​


​​ois.close();​​


​​} ​​ ​​catch​​ ​​(IOException e) {​​


​​e.printStackTrace();​​


​​}​​


​​}​​


​​}​​


​​}​​


你看,我们把一个对象存放在硬盘上是为了方便日后使用

现在用得着它了,自然得拿出来


java中的IO操作总结(四)_类对象_02

 

管道流

实例3:线程的通信

​​import​​          ​​java.io.IOException;​​        


​​import​​ ​​java.io.PipedInputStream;​​


​​import​​ ​​java.io.PipedOutputStream;​​





​​//实现Runnable接口,实现一个读的线程​​


​​class​​ ​​Read ​​ ​​implements​​ ​​Runnable {​​


​​private​​ ​​PipedInputStream in;​​


​​//将需要读的管道流传入到构造函数中​​


​​public​​ ​​Read(PipedInputStream in) {​​


​​this​​ ​​.in = in;​​


​​}​​





​​//实现读这一线程​​


​​public​​ ​​void​​ ​​run() {​​


​​try​​ ​​{​​


​​byte​​ ​​[] buf = ​​ ​​new​​ ​​byte​​ ​​[​​ ​​1024​​ ​​];​​


​​int​​ ​​temp = ​​ ​​0​​ ​​;​​


​​//循环读取​​


​​//read是一个阻塞方法,需要抛异常​​


​​//此处把打印流的代码也加入进来​​


​​//是因为如果没有读取到数据,那么打印的代码也无效​​


​​while​​ ​​((temp = in.read(buf)) != -​​ ​​1​​ ​​) {​​


​​String str = ​​ ​​new​​ ​​String(buf,​​ ​​0​​ ​​,temp);​​


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


​​}​​


​​} ​​ ​​catch​​ ​​(IOException e) {​​


​​//其实这里应抛出一个自定义异常的​​


​​//暂时我还没弄清楚​​


​​e.printStackTrace();​​


​​} ​​ ​​finally​​ ​​{​​


​​try​​ ​​{​​


​​//我已经抛火了,这只是为了提醒自己异常很重要​​


​​in.close();​​


​​} ​​ ​​catch​​ ​​(IOException e) {​​


​​e.printStackTrace();​​


​​}​​


​​}​​


​​} ​​


​​}​​





​​//这里实现一个写的类​​


​​class​​ ​​Write ​​ ​​implements​​ ​​Runnable {​​


​​private​​ ​​PipedOutputStream out;​​


​​//将管道输入流传进来​​


​​public​​ ​​Write(PipedOutputStream out) {​​


​​this​​ ​​.out = out;​​


​​}​​





​​public​​ ​​void​​ ​​run() {​​


​​try​​ ​​{​​


​​//这里开始写出数据​​


​​out.write(​​ ​​"管道输出"​​ ​​.getBytes());​​


​​} ​​ ​​catch​​ ​​(IOException e) {​​


​​e.printStackTrace();​​


​​} ​​ ​​finally​​ ​​{​​


​​try​​ ​​{​​


​​//其实应该可以把这个关闭方法写到上面那个try里边​​


​​//但是这样感觉怪怪的,逻辑不大对​​


​​out.close();​​


​​} ​​ ​​catch​​ ​​(IOException e) {​​


​​e.printStackTrace();​​


​​}​​


​​}​​


​​}​​


​​}​​





​​public​​ ​​class​​ ​​Demo {​​


​​public​​ ​​static​​ ​​void​​ ​​main(String[] args) {​​


​​PipedInputStream in = ​​ ​​new​​ ​​PipedInputStream();​​


​​PipedOutputStream out = ​​ ​​new​​ ​​PipedOutputStream();​​


​​try​​ ​​{​​


​​//连接管道​​


​​in.connect(out);​​





​​//创建对象,开启线程​​


​​//此处同样放进try...catch里面​​


​​//因为如果没有链接管道,下面操作无意义​​


​​Read r = ​​ ​​new​​ ​​Read(in);​​


​​Write w = ​​ ​​new​​ ​​Write(out);​​


​​//把已经实现好run方法的对象放入线程中执行​​


​​new​​ ​​Thread(r).start();​​


​​new​​ ​​Thread(w).start();​​





​​} ​​ ​​catch​​ ​​(IOException e) {​​


​​e.printStackTrace();​​


​​}​​


​​}​​


​​}​​


 好吧,废了那么大劲儿,就打印了这么一句话,异常抛弃来很烦,为了注重细节……


java中的IO操作总结(四)_序列化_03

管道流也许很难理解,其实非也

我们知道,字节流和字符流都需要数组来进行流的中转

而管道流则直接串联两条流,一边发送数据,一边接收

然而,同时通信的的两种状态,如何才能确定发送和接收的一致性呢

那么,就需要用到线程,无论是接收方还是发送方先执行

总会造成一个线程的阻塞状态,从而等待另一方的数据传过来

总体而言,管道流的目的,也就是为了线程通信

此外,还有PipedReader和PipedWriter类,操作原理都一样,这里就不再赘述了

 

DataOutputStream和DataInputStream类

实例4:基本数据类型的写入

​​import​​          ​​java.io.DataOutputStream;​​        


​​import​​ ​​java.io.File;​​


​​import​​ ​​java.io.FileOutputStream;​​


​​import​​ ​​java.io.IOException;​​





​​public​​ ​​class​​ ​​Demo {​​


​​public​​ ​​static​​ ​​void​​ ​​main(String[] args) {​​


​​String path = File.separator + ​​ ​​"home"​​ ​​+ File.separator + ​​ ​​"siu"​​ ​​+​​


​​File.separator + ​​ ​​"work"​​ ​​+ File.separator + ​​ ​​"demo.txt"​​ ​​;​​





​​DataOutputStream d = ​​ ​​null​​ ​​;​​


​​try​​ ​​{​​


​​//此处需要传入一个OutputStream类的对象​​


​​d = ​​ ​​new​​ ​​DataOutputStream(​​ ​​new​​ ​​FileOutputStream(path));​​


​​//开始写入基本数据类型​​


​​d.writeInt(​​ ​​12​​ ​​);​​


​​d.writeBoolean(​​ ​​true​​ ​​);​​


​​d.writeDouble(​​ ​​12.2223​​ ​​);​​


​​d.writeChar(​​ ​​97​​ ​​);​​


​​//刷新流​​


​​d.flush();​​





​​} ​​ ​​catch​​ ​​(IOException e) {​​


​​e.printStackTrace();​​


​​} ​​ ​​finally​​ ​​{​​


​​try​​ ​​{​​


​​d.close();​​


​​} ​​ ​​catch​​ ​​(IOException e) {​​


​​e.printStackTrace();​​


​​}​​


​​}​​


​​}​​


​​}​​


此处我们并不能直观看懂内容,因为它采用字节流的方式操作,而不是字符流

我们只需要知道,此程序已经将基本数据类型写入到硬盘即可


java中的IO操作总结(四)_java io 输入输出_04

 

实例5:基本数据类型的读取

​​import​​          ​​java.io.DataInputStream;​​        


​​import​​ ​​java.io.File;​​


​​import​​ ​​java.io.FileInputStream;​​


​​import​​ ​​java.io.IOException;​​





​​public​​ ​​class​​ ​​Demo {​​


​​public​​ ​​static​​ ​​void​​ ​​main(String[] args) {​​


​​String path = File.separator + ​​ ​​"home"​​ ​​+ File.separator + ​​ ​​"siu"​​ ​​+​​


​​File.separator + ​​ ​​"work"​​ ​​+ File.separator + ​​ ​​"demo.txt"​​ ​​;​​





​​DataInputStream d = ​​ ​​null​​ ​​;​​


​​try​​ ​​{​​


​​d = ​​ ​​new​​ ​​DataInputStream(​​ ​​new​​ ​​FileInputStream(path));​​


​​//按存储顺序读取基本数据类型​​


​​System.out.println(d.readInt());​​


​​System.out.println(d.readBoolean());​​


​​System.out.println(d.readDouble());​​


​​System.out.println(d.readChar());​​





​​} ​​ ​​catch​​ ​​(IOException e) {​​


​​e.printStackTrace();​​


​​} ​​ ​​finally​​ ​​{​​


​​try​​ ​​{​​


​​d.close();​​


​​} ​​ ​​catch​​ ​​(IOException e) {​​


​​e.printStackTrace();​​


​​}​​


​​}​​


​​}​​


​​}​​


 这里要注意的是,一定要按照写入顺序读取,否则会发生数据的打印错误


java中的IO操作总结(四)_java_05