java学习笔记:关于IO转换流InputStreamReader和OutputStreamWriter

1 字节流
InputStream                  ->FileInputStream
OutputStream                 ->FileOutputStream


2 字符流
Reader -> BufferedReader     -> LineNumberReader 
       -> InputStreamReader  -> FileReader

Writer -> BufferedWriter
       -> OutputStreamWriter -> FileWriter

============================================================
由JDK的IO流继承结构我们看到
FileInputStream和FileOutputStream是直接继承自字节流基类的

而FileReader和FileWriter是继承自2个转换流,
为什么会这样呢?
很可能是因为FileReader和FileWriter都是对字符型文件的操作,
而计算机处理的是字节数据,
读取文件后要将字符对象转成字节对象,
在写入到文件时时要将字节对象转成字符对象

 

 

 

 


 

/*
[示例1]:按行读写控制台中的字符
*/
import java.io.*;
class Demo            
{
  public static void main(String[] args)  throws IOException
  {
    readAndWriteCmd();
  }
 
  public static void readAndWriteCmd() throws IOException
  {
    //1.获取键盘输入的字节流对象
    InputStream in = System.in;
    
    //2.将字节流对象转成字符流对象,使用转换流InputStreamReader
    InputStreamReader isr = new InputStreamReader(in);  //in字节流就变成isr字符流了
    
    //3.为了提高效率,使用字符缓冲流BufferedReader
    BufferedReader bufr = new BufferedReader(isr);
    
    OutputStream output = System.out;  
    OutputStreamWriter osw = new OutputStreamWriter(output);  
    BufferedWriter bufw= new BufferedWriter(osw);  //因为要用到换行newLine()
    
    //4.然后就可以使用BufferedReader的readLine方法一次读取一行,读了再写到cmd里
    String lineStr=null;
    while(true)
    {
      lineStr = bufr.readLine();    //读取我们从键盘输入到控制台的内容
      if(lineStr!=null)
      {
        if(lineStr.equals("over"))   //输入over时结束
        {
          break;
        }
        else
        {
          bufw.write(lineStr); //写到控制台
          bufw.newLine();      //输出换行符,在windows里也可直接输出"\r\n"
          bufw.flush();        //清空缓冲区,否则下一次输出时会重复输出  
        }      
      }
      else
      {
        break;
      }
    }        
  }
}

 ========================================================================================

 

/*
将上个例子的输出设备改成硬盘文件

[示例2]:将键盘输入控制台的字符输出到文本文件
*/
import java.io.*;
class Demo            
{
  public static void main(String[] args)  throws IOException
  {
    readAndWriteCmd();
  }
 
  public static void readAndWriteCmd() throws IOException
  {
    BufferedReader bufr= new BufferedReader(new InputStreamReader(System.in));
    //BufferedWriter bufw= new BufferedWriter(new OutputStreamWriter(new FileOutputStream("c:\\out.txt"))); 
    BufferedWriter bufw= new BufferedWriter(new FileWriter("c:\\out.txt"));     
    String lineStr=null;
    while(true)
    {
      lineStr = bufr.readLine();    //读取我们从键盘输入到控制台的内容
      if(lineStr!=null)
      {
        if(lineStr.equals("over"))  //输入over时结束
        {
          break;
        }
        else
        {
          bufw.write(lineStr); 
          bufw.newLine();      //输出换行符,在windows里也可直接输出"\r\n"
          bufw.flush();        //清空缓冲区,否则下一次输出时会重复输出  
        }      
      }
      else
      {
        break;
      }
    }        
  }
}
 

当我们使用默认GBK编码保存文本时,下面2句代码其实是一样的效果,

new OutputStreamWriter(new FileOutputStream("out.txt"))

new FileWriter("out.txt")

 

当要求保存为其他编码比如UTF-8,就要这样写

new OutputStreamWriter(new FileOutputStream("out.txt"),"UTF-8")

 

而如果要读取一个UTF-8编码的文本文件时,同样的要用

new InputStreamReader(new FileInputStream("in.txt"),"UTF-8");

而不能用new FileWriter("in.txt")

 

============================================================================

笔记补充:

 

 

 

通过3个步骤来选择使用哪个流
1.明确源和目的体系
源:  输入流2个体系 InputStream Reader
目的:输出流2个体系 OutputStream Writer

2.操作的数据是否是纯文本
(为什么要区分字节和字符呢?效率问题,还有就是类中独有的方法)
是:用字符流
不是:字节流
对于键盘:我们一般要处理的是键盘录入的字符,如果你非要处理字节也有可能的

3.当体系明确后,再明确要使用哪个具体的类(或者说哪个对象)
通过设备来区分:
1.源设备 :内存.硬盘.键盘
2.目的设备:内存.硬盘.控制台

[针对示例2]:将键盘录入的字符数据,保存到硬盘文件中
首先看:键盘录入!
第一步:明确体系,键盘录入是源,用InputStream 或 Reader体系
第二步:操作的是否是纯文本? 键盘录入字符,所以是字符数据!用Reader
第三步:使用哪个具体对象? 我们知道键盘录入的对象使用System.in
这里我们可能有个问题要提出,
我们不是选择的Reader体系吗?而System.in是字节流啊!
System.in是java用来处理键盘录入的数据的,是处理的字节型数据没错,
但我们现在要处理的是字符问题,虽然字符本质上也是字节数据,但字符Reader体系
有它特有的方法并且处理字符时的高效便捷,所以我们才用Reader体系,
我们只要将获取的字节数据用Reader体系中的InputStreamReader转换一下就行了
InputStreamReader isr = new InputStreamReader(System.in);
另外: 需要提高效率吗? 需要,而且还有个输出换行符方法
BufferedReader bufr = new BufferedReader(isr);

再来看:保存文件!
第一步:明确体系,保存到硬盘是目的,是输出,用OutputStream或Writer
第二步:操作的是否是文本? 将字符写入到文件,是!用Writer字符流
第三步:使用哪个具体对象? 看设备,是硬盘,一个文件,使用FileWriter
FileWriter fw = new FileWriter("b.txt");
另外: 需要提高效率么?需要
BufferedWriter bufw = new BufferedWriter(fw);
这里我们不需要再使用转换流OutputStreamWriter
因为我们用的BufferedWriter包装的FileWriter,而FileWriter是继承OutputStreamWriter
的,BufferedReader 在写数据的时候就是写的字节数据

如果: 如果要求在保存文件时是使用指定的编码格式时,就要显式使用转换流OutputStreamWriter了
因为OutputStreamWriter的重载构造方法中有可以指定编码表名称的(默认编码表是GBK),
而其第一个参数又必须是字节输出流OutputStream, 而可以操作文件的字节输出流就是
FileOutputStream了.
需求:写入文本文件,编码格式是UTF-8
BufferedWriter bufw =
new BufferedWriter(new OutputStreamWriter(new FileOutpunStream("a.txt"),"UTF-8"))
而FileReader读取的是默认编码格式的文本文件,要读取UTF-8格式的文件,就要用到InputStreamReader

 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值