Java-IO:浅谈Java中的字节流和字符流

要把一片二进制数据数据逐一输出到某个设备中,或者从某个设备中逐一读取一片二进制数据,不管输入输出设备是什么,我们要用统一的方式来完成这些操作,用一种抽象的方式进行描述,即为IO流。

Java中的流按照操作单元可以分为两种,一种是字节流,另一种是字符流,分别由四个抽象类来表示(每种流包括输入和输出两种所以一共四个):InputStream,OutputStream,Reader(InputStreamReader),Writer(OutputStreamWriter)。Java中其他多种多样变化的流均是由它们派生出来的。

字节流对应的抽象类为OutputStream和InputStream ,不同的实现类就代表不同的输入和输出设备,它们都是针对字节进行操作的。

在应用中,经常要将完全是字符的一段文本输出去或读进来,用字节流可以吗?计算机中的一切最终都是二进制的字节形式存在。对于“中国”这些字符,首先要得到其对应的字节,然后将字节写入到输出流。读取时,首先读到的是字节,可是我们要把它显示为字符,我们需要将字节转换成字符。由于这样的需求很广泛,于是提供了字符流的包装类。底层设备永远只接受字节数据,有时候要写字符串到底层设备,需要将字符串转成字节再进行写入。字符流是字节流的包装,字符流则是直接接受字符串,它内部将串转成字节,再写入底层设备,这为我们向IO设备写入或读取字符串提供了方便。

字符向字节转换时,要注意编码的问题,因为字符串转成字节数组,其实是转成该字符的某种编码的字节形式,读取也是反之的道理。

字符流和字节流是根据处理数据的不同来区分的。字节流按照8位传输,字符流按照16位传输。字节流是最基本的,所有文件的储存是都是字节(byte)的储存,在磁盘上保留的并不是文件的字符而是先把字符编码成字节,再储存这些字节到磁盘。

  • 1.字节流可用于任何类型的对象,包括二进制对象,而字符流只能处理字符或者字符串;
  • 2.字节流提供了处理任何类型的IO操作的功能,但它不能直接处理Unicode字符,而字符流就可以。

读文本的时候用字符流,例如txt文件。读非文本文件的时候用字节流,例如mp3。理论上任何文件都能够用字节流读取,但当读取的是文本数据时,为了能还原成文本你必须再经过一个转换的工序,相对来说字符流就省了这个麻烦,可以有方法直接读取。

字符流处理的单元为2个字节的Unicode字符,分别操作字符、字符数组或字符串,而字节流处理单元为1个字节, 操作字节和字节数组。所以字符流是由Java虚拟机将字节转化为2个字节的Unicode字符为单位的字符而成的,所以它对多国语言支持性比较好!

1.字节流:继承于InputStream\OutputStream

OutputStream提供的方法:

写入一个字节的数据
void write(int b);

将数组buffer的数据写入流
void write(byte[] buffer);

从buffer[offset]开始,写入len个字节的数据
void write(byte[] buffer,int offset,int len);

强制将buffer内的数据写入流
void flush();

关闭流 
void close();

InputStream提供的方法:

读出一个字节的数据,如果已达文件的末端,返回值为-1
int read();

读出buffer大小的数据,返回值为实际所读出的字节数
int read(byte[] buffer);
int read(byte[] buffer,int offset,int len)

返回流内可供读取的字节数目
int available();

跳过n个字节的数据,返回值为实际所跳过的数据数
long skip(long n);

关闭流 
void close();

2.字符流,继承于InputStreamReader\OutputStreamWriter

字符流的类:

1),BufferedReader是一种过滤器(filter),继承FilterReader。过滤器用来将流的数据加以处理再输出。

构造函数为:

生成一个缓冲的字符输入流,in为一个读取器
BufferedReader(Reader in);

生成一个缓冲的字符输入流,并指定缓冲区的大小为size
BufferedReader(Reader in,int size);

示例:IOStreamDemo:

import java.io.*;

public class IOStreamDemo {

        public void samples() throws IOException { 
            //1. 这是从键盘读入一行数据,返回的是一个字符串           
            BufferedReader stdin =new BufferedReader(new InputStreamReader(System.in));                  
            System.out.print("Enter a line:");            
            System.out.println(stdin.readLine());

            //2. 这是从文件中逐行读入数据
            BufferedReader in = new BufferedReader(new FileReader("IOStreamDemo.java"));                  
            String s, s2 = new String();            

            while((s = in.readLine())!= null){   
                s2 += s + "\n";
            }
                                                  
            in.close();

            //3. 这是从一个字符串中逐个读入字节            
            StringReader in1 = new StringReader(s2);           

            int c;            
            while((c = in1.read()) != -1) {   
                System.out.print((char)c);  
            }

            //4. 这是将一个字符串写入文件            
            try {
                BufferedReader in2 = new BufferedReader(new StringReader(s2));                                        
                PrintWriter out1 = new PrintWriter(new BufferedWriter(new FileWriter("IODemo.out")));                       

                int lineCount = 1;                       
                while((s = in2.readLine()) != null )    {
                    out1.println(lineCount++ + ": " + s);
                }                                                    

                out1.close();           
            } catch(EOFException e) {
                System.err.println("End of stream");           
            }
        }

}

对于上面的例子,需要说明的有以下几点:

1. InputStreamReader是InputStream和Reader之间的桥梁,由于System.in是字节流,需要用它来包装之后变为字符流供给BufferedReader使用。

2. PrintWriter out1 = new PrintWriter(new BufferedWriter(new FileWriter("IODemo.out")));

这句话体现了Java输入输出系统的一个特点,为了达到某个目的,需要包装好几层。首先,输出目的地是文件IODemo.out,所以最内层包装的是FileWriter,建立一个输出文件流,接下来,我们希望这个流是缓冲的,所以用BufferedWriter来包装它以达到目的,最后,我们需要格式化输出结果,于是将PrintWriter包在最外层。

  • 5
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

王小二(海阔天空)

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值