2.字节流
|-- InputStream(读)
|-- OutputStream(写)
由于字节是二进制数据,所以字节流可以操作任何类型的数据,值得注意的是字符流使用的是字符数组char[]而字节流使用的是字节数组byte[]。下面来看一个字节流读写文件的简单例子。
清单7,使用字节流读写文本文件代码
private static void test5(){
FileOutputStream fos=null;
try{
fos=new FileOutputStream("D:/test.txt");
fos.write(0010);//写入二进制数据
fos.flush();
}catch(IOException e){
}finally{
try{
fos.close();
}catch(IOException ex){
}
}
FileInputStream fis=null;
try{
fis=new FileInputStream("D:/test.txt");
//fis.available()是获取关联文件的字节数,即test.txt的字节数
//这样创建的数组大小就和文件大小刚好相等
//这样做的缺点就是文件过大时,可能超出jvm的内存空间,从而造成内存溢出
byte[] buf=new byte[fis.available()];
fis.read(buf);
System.out.println(new String(buf));
}catch(IOException e){
}finally{
try{
fos.close();
}catch(IOException ex){
}
}
}
清单8,使用缓冲区对一张图片进行复制代码
private static void test6(){
BufferedOutputStream bos=null;
BufferedInputStream bis=null;
try{
//前面已经说过了,缓冲对象是根据具体的流对象创建的,所以必须要有流对象
bis=new BufferedInputStream(new FileInputStream("E:\images\wo\1.jpg"));
//写入目标地址
bos=new BufferedOutputStream(new FileOutputStream("E:\test.jpg"));
byte[] buf=new byte[1024];
while((bis.read(buf))!=-1){
bos.write(buf);
}
bos.flush();
}catch(IOException e){
e.toString();
}finally{
try{
if(bos!=null){
bos.close();
}
if(bis!=null){
bis.close();
}
}catch(IOException ex){
ex.toString();
}
}
}
3.转换流
特点
|--是字节流和字符流之间的桥梁
|--该流对象可以对读取到的字节数据进行指定编码表的编码转换
何时使用
|--当字节和字符之间有转换动作时
|--流操作的数据需要进行编码表的指定时
具体对象体现
|--InputStreamReader:字节到字符的桥梁
|--OutputStreamWriter:字符到字节的桥梁
说明
这两个流对象是字符流体系中的成员,它们有转换的作用,而本身又是字符流,
所以在new的时候需要传入字节流对象。
构造函数
|--InputStreamReader(InputStream)
通过该构造函数初始化,使用的是系统默认的编码表GBK。
|--InputStreamReader(InputStream,String charset)
通过该构造函数初始化,可以通过charset参数指定编码。
|--OutputStreamWriter(OutputStream)
使用的是系统默认的编码表GBK。
|--OutputStreamWriter(OutputSream,String charset)
通过该构造函数初始化,可以通过参数charset指定编码。
操作文件的字符流对象是转换流的子类
|--Reader
|--InputStreamReader(转换流)
|--FileReader(文件字符流)
|--Writer
|--OutputStreamWriter(转换流)
|--FileWriter(文件字符流)
说明
转换流中的read方法,已经融入了编码表,在底层调用字节流的
read方法时将获取的一个或者多个字节数据进行临时存储,并去
查指定的编码表,如果编码没有指定,则使用默认编码表。
既然转换流已经完成了编码转换的动作,对于直接操作的文本文
件的FileReader而言,就不用再重新定义了,只要继承该转换
流,获取其方法,就可以直接操作文本文件中的字符数据了。
注意
在使用FileReader操作文本数据时,该对象使用的是默认的编码表,
如果要使用指定的编码表,必须使用转换流。
代码体现
FileReader fr=new FileReader(“test.txt”);
InputStreamReader isr=new InputStreamReader(new
FileInputStreamReader(“test.txt”));
这两句代码意义相同,操作test.txt中的数据都是使用了系统默认
的编码GBK。因为我们系统默认使用的编码表是GBK,如果test.txt中
的数据是通过UTF-8形式编码的,那么在读取的时候就需要指定编码表,
因此转换流必须使用InputStreamReader isr=new
InputStreamReader(new FileInputStream(“a.txt”),”UTF-8”);
四、流操作的基本规律
|--明确数据源和数据汇(数据目的)
其实是为了明确是输入流还是输出流
|--明确操作的数据是否是纯文本数据
|--说明
数据源
键盘System.in、硬盘、File开头的流对象、内存(数组)。
数据汇
控制台System.out、硬盘、File开头的流对象、内存(数组)。
|--需求
将键盘录入的数据存储到一个文件中和打印到控制台
|--数据源System.in
既然是源,使用的就是输入流,可用的体系有InputStream
、Reader。因为键盘录入进来的一定是纯文本数据,所以
可以使用专门操作字符数据的Reader。而System.in对应
的流是字节读取流,所以要将其进行转换,将字节转换成字
符即可,所以要使用Reader体系中的InputStreamReader,
如果要提高效率,就使用BufferedReader,代码如:
BufferedReader bur=new BufferedReader(new
InputStreamReader(Sysem.in));
|--数据汇:一个文件、硬盘
数据汇一定是输出流,可以用的体系有OutputStream、
Writer。往文件中存储的都是文本数据,那么可以使用字符
流较为方便Writer。因为操作的是一个文件,所以使用Writer
中的FileWriter,同理,要提高效率就要使用BufferedWriter。
代码如:BufferedWriter bufr=new BufferedWriter(new
FileWriter(“test.txt”));
清单9,将键盘录入的数据存储到一个文件中和打印到控制台代码
private static void test7(){
BufferedReader bur=null;
OutputStreamWriter osw=null;
BufferedWriter bw=null;
try{
//数据源
bur=new BufferedReader(new InputStreamReader(System.in));
//数据汇
osw=new OutputStreamWriter(System.out);
//数据汇,因为数据源用的是系统默认编码,所以这里可以直接使用FileWriter
//否则必须使用OutputStreamWriter转换流
bw=new BufferedWriter(new FileWriter("D:\test_target.txt"));
String line=null;
while((line=bur.readLine())!=null){
osw.write(line);
osw.flush();//刷新到控制台
bw.write(line);
bw.flush();//刷新到文本文件
}
}catch(IOException e){
e.toString();
}finally{
try{
if(osw!=null){
osw.close();
}
if(bur!=null){
bur.close();
}
if(bw!=null){
bw.close();
}
}catch(IOException ex){
ex.toString();
}
}
}
清单9是按照默认编码表写入文本文件的,那么如何按照指定编码表写入文件呢?其实也很简单,将清单9的代码稍微改一下就可以了。
清单10代码
private static void test8(){
BufferedReader bur=null;
BufferedWriter bw=null;
try{
//数据源
bur=new BufferedReader(new InputStreamReader(System.in));
//数据汇,按照指定编码格式存储到文本文件
bw=new BufferedWriter(new OutputStreamWriter(new
FileOutputStream("D:\test_target.txt"),"UTF-8"));
String line=null;
while((line=bur.readLine())!=null){
bw.write(line);
bw.flush();//刷新到文本文件
}
}catch(IOException e){
e.toString();
}finally{
try{
if(bur!=null){
bur.close();
}
if(bw!=null){
bw.close();
}
}catch(IOException ex){
ex.toString();
}
}
}
既然写入文件时指定了编码,那么在读取的时候就必须指定该编码才能正确显示。
清单11,读取指定编码表的文件代码
private static void test9() {
BufferedReader bur = null;
try {
// 注意,这里读取的是清单8写入的文件,
// 清单10用UTF-8编码格式写入,
// 所以在构造InputStreamReader时必须指定UTF-8编码
bur = new BufferedReader(new InputStreamReader(
new FileInputStream("D:\test_target.txt"), "UTF-8"));
String line = null;
while ((line = bur.readLine()) != null) {
System.out.println(line);
}
} catch (IOException e) {
e.toString();
} finally {
try {
if (bur != null) {
bur.close();
}
} catch (IOException ex) {
ex.toString();
}
}
}
写入和读取都做了,现在还差个复制操作,其实复制文件也很简单,先读取文件,再将读取到的数据写入文件,不同的是,在读取和写入时我们可以指定编码表。
清单12代码
private static void test11() {
BufferedReader bur = null;
BufferedWriter buw = null;
try {
bur = new BufferedReader(new InputStreamReader(
new FileInputStream("D:\test_target.txt"), "UTF-8"));
buw = new BufferedWriter(new OutputStreamWriter(
new FileOutputStream("D:\test_target1.txt"),"UTF-8"));
String line = null;
while ((line = bur.readLine()) != null) {
buw.write(line);
buw.flush();// 刷新到文本文件
}
} catch (IOException e) {
e.toString();
} finally {
try {
if (buw != null) {
buw.close();
}
if (bur != null) {
bur.close();
}
} catch (IOException ex) {
ex.toString();
}
}
}