10、IO中的其他流:
(1)打印流:
**PrintWriter:字符打印流
****构造方法:
PrintWriter(String fileName)
创建具有指定文件名称且不带自动行刷新的新 PrintWriter。
PrintWriter(File file)
使用指定文件创建不具有自动行刷新的新 PrintWriter。
PrintWriter(Writer out)
创建不带自动行刷新的新 PrintWriter。
PrintWriter(Writer out, boolean autoFlush)
自动刷新
PrintWriter(OutputStream out)
根据现有的 OutputStream 创建不带自动行刷新的新 PrintWriter。
PrintWriter(OutputStream out, boolean autoFlush)
自动刷新
****方法摘要:
PrintWriter append(char c)
将指定字符添加到此 writer。
void close()
关闭该流并释放与之关联的所有系统资源。
void flush()
刷新该流的缓冲。
void print(Object obj)
打印对象。
void print(String s)
打印字符串。
void println()
通过写入行分隔符字符串终止当前行。
**PrintStream:字节打印流
****构造方法:
PrintStream(String fileName)
创建具有指定文件名称且不带自动行刷新的新打印流。
PrintStream(File file)
创建具有指定文件且不带自动行刷新的新打印流。
PrintStream(OutputStream out)
创建新的打印流。
PrintStream(OutputStream out, boolean autoFlush)
创建新的打印流。
****方法摘要:
PrintWriter append(char c)
将指定字符添加到此 writer。
void close()
关闭该流并释放与之关联的所有系统资源。
void flush()
刷新该流的缓冲。
void print(Object obj)
打印对象。
void print(String s)
打印字符串。
void println()
通过写入行分隔符字符串终止当前行。
(2)对象系列化:
**对象实体化:找一个介质,能长期的存储对象。
**对象的属性在Java程序中,都是存在于对内存中,随着对象的消失而消失,
而ObjectOutputStream可以将对象实体化
**Serializable接口没有一个方法,也就是说其是一个标记接口。比如盖章的猪肉才是安全的。
**只有实现Serializable接口的子类才能被ObjectOutputStream系列化写入流,当某个
类实现该接口后,会被Java自动分配UID号,以便编译器识别,区分不同对象。
**用ObjectOutputStream系列化的对象存储到文件后,该文件是乱码,也就是不可读的
的用ObjectInputStream读取该类对象的属性。
**由于对象是有Java给对象分配相应的UID号,而UID号是根据对象的属性不同而分配的。
当一个类对象被系列化到文件后,如果该类改动了对象的属性,比如将某个成员变量变成私有
则该对象再用ObjectInputStream读取时会报异常,也就是说该系列化到文件的对象不能再被使用了
那么,要想继续使用属性被改动后的对象,我们可以自定义给对象分配UID号,让UID号不随对象的属性
变化而变化。
自定义对象分配UID方法如下:
public static final long serialVersion UID = 43L;
**注意:
静态不能被系列化,因为静态成员变量实在内存的方法区,而ObjectOutputStream只能
对对内存里面的数据进行系列化
被transient修饰的非静态成员变量也不能被系列化
被系列化的对象存储到文件中,该文件是不可读的,所以该文件的扩展名一般
不写成.txt,通常后缀名写.object
**ObjectOutputStream
**ObjectInputStream
(3)管道流:
PipedInputStream
PipedOutputStream
(4)随机访问文件:RandomAccess(重要!!!)
**自身具备读写方法(很牛逼!又可以读又可以写)
**通过skipByte(int x)和seek(int x)来达到随机访问文件
**该类不是IO体系子类,而是直接继承Object,但它是IO包中的成员,因为它具备读写方法
**该类内部封装了数组,而且通过指针对数组的元素进行操作,可以通过getFilePoint获取指针位置
同时可以通过seek改变指针位置
**该类完成读写的原理是内部封装了字节输入输出流
**通过该类的构造看出,该类只能操作文件,而且操作的文件只能有固定模式:
"r":只读
"rw":读写
"rws":
"red":
**构造方法:
RandomAccessFile(File file, String mode)
创建从中读取和向其中写入(可选)的随机访问文件流,该文件由 File 参数指定。
RandomAccessFile(String name, String mode)
创建从中读取和向其中写入(可选)的随机访问文件流,该文件具有指定名称。
**方法摘要:
void write(byte[] b)
将 b.length 个字节从指定 byte 数组写入到此文件,并从当前文件指针开始。
void write(byte[] b, int off, int len)
将 len 个字节从指定 byte 数组写入到此文件,并从偏移量 off 处开始。
void write(int b)
向此文件写入指定的字节。
int read()
从此文件中读取一个数据字节。
int read(byte[] b)
将最多 b.length 个数据字节从此文件读入 byte 数组。
int read(byte[] b, int off, int len)
将最多 len 个数据字节从此文件读入 byte 数组。
String readLine()
从此文件读取文本的下一行。
long getFilePointer()
返回此文件中的当前偏移量。
long length()
返回此文件的长度。
void seek(long pos)
设置到此文件开头测量到的文件指针偏移量,在该位置发生下一个读取或写入操作。
(4)操作基本数据类型的流对象:DateStream
(5)操作字节数组流:
ByteArrayInputStream
ByteArrayOutputStream
11、IO流转换流的字符编码
(1)字符流的出现为了方便操作字符,更重要的是加入了编码转换
(2)通过子类转换流来完成
InputStreamReander
OutputStreamWriter
(3)在两个子类对象进行构造的时候可以加入编码表
(4)编码表:
将各个国家的文字用二进制数字表示并一一对应,形成一张表,这就是编码表
(5)常见的编码表:
**ASCII:美国标准信息交换码,用一个字节的七位表示
**ISO8859-1:拉丁码表,欧洲码表,用一个字节的八位表示
**GB2312:中文编码表,用两个字节表示
**GBK:中文编码表升级,融合录入更多的中文字符,用两个字节表示,为避免和老美重复
两字节的最高位都是1,即汉字都是用负数表示
**Unicode:国际标准码,融合了多种文字,所有文字都用两个字节表示
**UTF-8:用一个字节到三个字节表示。
注:Unicode能识别中文,UTF-8也能识别中文,但两种编码表示一个汉字所用的字节数不同
Unicode用两个字节,UTF-8用三个字节,故涉及到编码转换。
(6)在流中涉及编码表的转换只有转换流:
InputStreamReander
OutputStreamWriter
(7)代码示例:
public static void write() throws IOException
{
OutputStreamWriter osw1 = new OutputStreamWriter(new FileOutputStream("gbk.txt"),"GBK");
osw1.write("你好");
osw1.close();
OutputStreamWriter osw2 = new OutputStreamWriter(new FileOutputStream("utf-8.txt"),"UTF-8");
osw2.write("你好");
osw2.close();
}
public static void read() throws IOException
{
InputStreamReader isr = new InputStreamReader(new FileInputStream("gbk.txt"),"GBK");
byte[] buf = new byte[1024];
int len = isr.read(buf);
sop(new String(buf,0,len));
}
(8)编码解码
编码:字符串变成字节数组:String-->getBytes()-->byte[]()
解码:字节数组变成字符串:byte[]-->new String(byte[],0,len)-->String
(9)代码示例:
public static void main(String[] args)
{
//编码解码1:默认编码
String str1 = "你好";
byte[] buf1 = str1.getBytes();//默认解码:Unicode,四个字节
//编码解码2:指定编码
String str2 = "你好";
byte[] buf2 = str2.getBytes("UTF-8");//指定解码:UTF-8,六个字节
//编码解码3:编码正确解码错误
String str3 = "你好";
byte[] buf3 = str3.getBytes("GBK");//指定编码:GBK,四个字节
String str3 = new String(buf3,"ISO8859-1");//错误解码
//编码解码4:错误编码正确解码
String str4 = "你好";
byte[] buf4 = str4.getBytes("ISO8859-1");//错误编码
String str4 = new String(buf4,"GBK");//正确解码,读不出来
//编码解码5:编码对了,但是解码错误了,怎么办呢?
//此时可以将错误的解码再错编回去,载用正确编码解码
String str5 = "你好";
byte[] buf5 = str5.getBytes("GBK");//正确编码
String str6 = new String(buf5,"ISO8859-1");//错误解码,读不出来
byte[] buf6 = str6.getBytes("ISO8859-1");//再错误编码
String str7 = new String(buf6,"GBK");//再正确解码,这样就可以读出来了
}
|