IO编程总结

文件传输基础----Java IO流

java.io.File类用于表示文件(目录)
File类只用于表示文件(目录)的信息(名称、大小等),不能用于文件内容的访问

RandomAccessFile java提供的对文件内容的访问,既可以读文件,页可以写文件
RandomAccessFile 支持随机访问文件,可以访问文件的任意位置

1)java文件模型

在硬盘上的文件时byte byte byte存储的,是数据的集合

2)打开文件

有两种模式“rw”(读写),“r”(只读)
RandomAccessFile raf = new RandomAccessFile(file, "rw")
文件指针,打开文件时指针在开头 pointer = 0;

3)写方法

raf.write(int)--->只写一个字节(后8位),同时指针指向像一个位置,准备再次写入

4)读方法

int b = raf.read()--->读一个字节

5)文件读写完成以后一定要关闭(Oracle官方说明)

IO流(输入流、输出流)

字节流、字符流

1.字节流

1)InputStream和OutputStream

InputStream抽象了应用程序读取数据的方式
OutputStream抽象了应用程序写出数据的方式

2)EOF = End 读到-1就读到了结尾

3)输入流基本方法

int b = in.read(); 读取一个字节,无符号填充到int的低8位. -1是EOF
in.read(byte[] buf) 读取数据填充到字节数组buf
in.read(byte[] buf, int start, int size) 读取数据到字节数组buf,从buf的start位置开始,存放size长度的数组

4)输出流基本方法

out.write(int b) 写出一个byte到流,b的低8位
out.write(byte[] buf) 将buf字节数组都写入到流
out.write(byte[] buf, int start, int size)

5)FileInputStream--->继承了InputStream,具体实现了在文件上读取数据

6)FileOutputStream--->实现了向文件中写出byte数据的方法

7)DataOutputStream/DataInputStream

对“流”功能的扩展,可以更加方便的读取int,long,字符等类型的数据
DataOutputStream
writeInt()/writeDouble()/writeUTF() (装饰模式实现的,将功能包装好)

8)字节缓冲流 BufferedInputStream & BufferedOutputStream

这两个流类为IO提供了带缓冲区的操作,一般打开文件进行写入或读取操作时,都会加上缓冲,这种流模式提高了IO的性能
从应用程序中把输入放入文件,相当于将一缸水导入到另一个缸中:
FileOutputStream--->write()方法相当于一滴一滴地把水转移过去
DataFileOutputStream--->writeXxx()方法会方便一些,相当于一瓢一瓢地把水转移过去
BufferedOutputStream--->write方法更方便,相当于一瓢一瓢先放入桶(相当于缓冲区)中,再从桶中倒入缸中,性能提高

2.字符流

1)编码问题

2)认识文本和文本文件

java的文本(char)时16位无符号整数,时字符的unicode编码(双字节编码)
文件是byte byte byte... 的数据序列
文本文件时文本(char)序列按照某种编码方案(utf-8\utf-16be\gbk)序列化位byte的存储结果

3)字符流(Reader Writer)

字符的处理,一次处理一个字符
字符的底层仍然是基本的字节序列
字符流的基本实现
InputStreamReader 完成byte流解析位char流,按照编码解析
OutputStreamWriter 提供char流到byte流,按照编码处理
FileReader/FileWriter
字符流的过滤器
BufferedReader ----> readLine 一次读一行
BufferedWriter/PrintWriter 写一行

3. 对象的序列化和反序列化

1)对象的序列化:就是将Object对象转换成byte序列,反之叫对象的反序列化

2)序列化流(ObjectOutputStream),是过滤流---writeObject

反序列化流(ObjectInputStream),---readObject

3)序列化接口(Serializable)

对象必须实现序列化接口,才能进行序列化,否则将出现异常
这个接口,没有任何方法,只是一个标准

若希望对象在网络中传播,则需要将文件转换成字节序列
用反序列化的方式将序列化的字节序列读取出来
在网络空间中有些对象是不需要序列化的,但是默认的序列化可能会占用大量的空间,因此使用transient进行修饰,不让其做默认的序列化

4)transient关键字(可以提高性能)

private void writeObject(java.io.ObjectOutputStream s) //方法签名
throws java.io.IOException{
s.defaultWriteObject(); //把jvm能默认序列化的元素进行序列化操作
s.writeInt(stuage); //自己完成stuage的序列化
}

private void readObject(java.io.ObjectInputStream s)
throws java.io.IOException, ClassNotFoundException{
s.defaultReadObject(); //把jvm能默认反序列化的元素进行反序列化操作
this.stuage = s.readInt(); //自己完成stuage的反序列化的操作
}
分析ArrayList源码中序列化和反序列化的问题:将数组中的有效元素进行序列化和反序列化

5)序列化中子父类构造函数问题

     * 对子类对象进行反序列化操作时,如果其父类面诶呦实现序列化接口
     * 那么其父类的构造函数会被调用
     * 如果实现了序列化接口,其父类的构造函数不会被调用(在保存的文件中就可以读到,不需要调用)
posted @ 2020-03-13 10:52  liujy1  阅读(334)  评论(0编辑  收藏  举报