`

字节流与字符流的区别详解

 
阅读更多

先来看一下流的概念:

在程序中所有的数据都是以流的方式进行传输或保存的,程序需要数据的时候要使用输入流读取数据,而当程序需要将一些数据保存起来的时候,就要使用输出流完成。

程序中的输入输出都是以流的形式保存的,流中保存的实际上全都是字节文件。

字节流与字符流

java.io包中操作文件内容的主要有两大类:字节流、字符流,两类都分为输入和输出操作。在字节流中输出数据主要是使用OutputStream完成,输入使的是InputStream,在字符流中输出主要是使用Writer类完成,输入流主要使用Reader类完成。(这四个都是抽象类)

java中提供了专用于输入输出功能的包Java.io,其中包括:
InputStream,OutputStream,Reader,Writer
InputStream 和OutputStream,两个是为字节流设计的,主要用来处理字节或二进制对象,
Reader和 Writer.两个是为字符流(一个字符占两个字节)设计的,主要用来处理字符或字符串.


字符流处理的单元为2个字节的Unicode字符,分别操作字符、字符数组或字符串,而字节流处理单元为1个字节,操作字节和字节数组。所以字符流是由Java虚拟机将字节转化为2个字节的Unicode字符为单位的字符而成的,所以它对多国语言支持性比较好!如果是音频文件、图片、歌曲,就用字节流好点,如果是关系到中文(文本)的,用字符流好点
所有文件的储存是都是字节(byte)的储存,在磁盘上保留的并不是文件的字符而是先把字符编码成字节,再储存这些字节到磁盘。在读取文件(特别是文本文件)时,也是一个字节一个字节地读取以形成字节序列

字节流可用于任何类型的对象,包括二进制对象,而字符流只能处理字符或者字符串; 2. 字节流提供了处理任何类型的IO操作的功能,但它不能直接处理Unicode字符,而字符流就可以
字节流是最基本的,所有的InputStrem和OutputStream的子类都是,主要用在处理二进制数据,它是按字节来处理的 但实际中很多的数据是文本,又提出了字符流的概念,它是按虚拟机的encode来处理,也就是要进行字符集的转化 这两个之间通过 InputStreamReader,OutputStreamWriter来关联,实际上是通过byte[]和String来关联 在实际开发中出现的汉字问题实际上都是在字符流和字节流之间转化不统一而造成的

==================我们还可以看到:============
Reader类的read()方法返回类型为int :作为整数读取的字符(占两个字节共16位),范围在0 到 65535之间 (0x00-0xffff),如果已到达流的末尾,则返回 -1


inputStream的read()虽然也返回int,但由于此类是面向字节流的,一个字节占8个位,所以返回0 到 255范围内的 int 字节值。如果因为已经到达流末尾而没有可用的字节,则返回值 -1。因此对于不能用0-255来表示的值就得用字符流来读取!比如说汉字.


操作流程

JavaIO操作也是有相应步骤的,以文件操作为例,主要的操作流程如下:

1使用File类打开一个文件

2通过字节流或字符流的子类,指定输出的位置

3进行读/写操作

4关闭输入/输出

IO操作属于资源操作,一定要记得关闭

字节流

字节流主要是操作byte类型数据,以byte数组为准,主要操作类就是OutputStreamInputStream

字节输出流:OutputStream

OutputStream是整个IO包中字节输出流的最大父类,此类的定义如下:

publicabstractclassOutputStreamextendsObjectimplementsCloseable,Flushable

从以上的定义可以发现,此类是一个抽象类,如果想要使用此类的话,则首先必须通过子类实例化对象,那么如果现在要操作的是一个文件,则可以使用:FileOutputStream类。通过向上转型之后,可以为OutputStream实例化

Closeable表示可以关闭的操作,因为程序运行到最后肯定要关闭

Flushable:表示刷新,清空内存中的数据

FileOutputStream类的构造方法如下:

publicFileOutputStream(Filefile)throwsFileNotFoundException

写数据:

1importjava.io.File;
2importjava.io.FileOutputStream;
3importjava.io.IOException;
4importjava.io.OutputStream;
5
6publicclassTest11{
7publicstaticvoidmain(String[]args)throwsIOException{
8Filef=newFile("d:"+File.separator+"test.txt");
9OutputStreamout=newFileOutputStream(f);//如果文件不存在会自动创建
10Stringstr="HelloWorld";
11byte[]b=str.getBytes();
12out.write(b);//因为是字节流,所以要转化成字节数组进行输出
13out.close();
14}
15}

也可以一个字节一个字节进行输出,如下:

1importjava.io.File;
2importjava.io.FileOutputStream;
3importjava.io.IOException;
4importjava.io.OutputStream;
5
6publicclassTest11{
7publicstaticvoidmain(String[]args)throwsIOException{
8Filef=newFile("d:"+File.separator+"test.txt");
9OutputStreamout=newFileOutputStream(f);//如果文件不存在会自动创建
10Stringstr="HelloWorld";
11byte[]b=str.getBytes();
12for(inti=0;i<b.length;i++){
13out.write(b[i]);
14}
15out.close();
16}
17}

以上输出只会进行覆盖,如果要追加的话,请看FileOutputStream类的另一个构造方法:

publicFileOutputStream(Filefile,booleanappend)throwsFileNotFoundException

在构造方法中,如果将append的值设置为true,则表示在文件的末尾追加内容。

1importjava.io.File;
2importjava.io.FileOutputStream;
3importjava.io.IOException;
4importjava.io.OutputStream;
5
6publicclassTest11{
7publicstaticvoidmain(String[]args)throwsIOException{
8Filef=newFile("d:"+File.separator+"test.txt");
9OutputStreamout=newFileOutputStream(f,true);//追加内容
10Stringstr="\r\nHelloWorld";
11byte[]b=str.getBytes();
12for(inti=0;i<b.length;i++){
13out.write(b[i]);
14}
15out.close();
16}
17}

文件中换行为:\r\n

字节输入流:InputStream

既然程序可以向文件中写入内容,则就可以通过InputStream从文件中把内容读取进来,首先来看InputStream类的定义:

publicabstractclassInputStreamextendsObjectimplementsCloseable

OutputStream类一样,InputStream本身也是一个抽象类,必须依靠其子类,如果现在是从文件中读取,就用FileInputStream来实现。

观察FileInputStream类的构造方法:

publicFileInputStream(Filefile)throwsFileNotFoundException

读文件:

1importjava.io.File;
2importjava.io.FileInputStream;
3importjava.io.IOException;
4importjava.io.InputStream;
5
6publicclassTest12{
7publicstaticvoidmain(String[]args)throwsIOException{
8Filef=newFile("d:"+File.separator+"test.txt");
9InputStreamin=newFileInputStream(f);
10byte[]b=newbyte[1024];
11intlen=in.read(b);
12in.close();
13System.out.println(newString(b,0,len));
14}
15}

但以上方法是有问题的,用不用开辟这么大的一个字节数组,明显是浪费嘛,我们可以根据文件的大小来定义字节数组的大小,File类中的方法:publiclonglength()

1importjava.io.File;
2importjava.io.FileInputStream;
3importjava.io.IOException;
4importjava.io.InputStream;
5
6publicclassTest13{
7publicstaticvoidmain(String[]args)throwsIOException{
8Filef=newFile("d:"+File.separator+"test.txt");
9InputStreamin=newFileInputStream(f);
10byte[]b=newbyte[(int)f.length()];
11in.read(b);
12in.close();
13System.out.println(newString(b));
14}
15}

我们换种方式,一个字节一个字节读入~

1importjava.io.File;
2importjava.io.FileInputStream;
3importjava.io.IOException;
4importjava.io.InputStream;
5
6publicclassTest14{
7publicstaticvoidmain(String[]args)throwsIOException{
8Filef=newFile("d:"+File.separator+"test.txt");
9InputStreamin=newFileInputStream(f);
10byte[]b=newbyte[(int)f.length()];
11for(inti=0;i<b.length;i++){
12b[i]=(byte)in.read();
13}
14in.close();
15System.out.println(newString(b));
16}
17}

但以上情况只适合知道输入文件的大小,不知道的话用如下方法:

1importjava.io.File;
2importjava.io.FileInputStream;
3importjava.io.IOException;
4importjava.io.InputStream;
5
6publicclassTest15{
7publicstaticvoidmain(String[]args)throwsIOException{
8Filef=newFile("d:"+File.separator+"test.txt");
9InputStreamin=newFileInputStream(f);
10byte[]b=newbyte[1024];
11inttemp=0;
12intlen=0;
13while((temp=in.read())!=-1){//-1为文件读完的标志
14b[len]=(byte)temp;
15len++;
16}
17in.close();
18System.out.println(newString(b,0,len));
19}
20}


字符流

在程序中一个字符等于两个字节,那么java提供了ReaderWriter两个专门操作字符流的类。


字符输出流:Writer

Writer本身是一个字符流的输出类,此类的定义如下:

publicabstractclassWriterextendsObjectimplementsAppendableCloseableFlushable

此类本身也是一个抽象类,如果要使用此类,则肯定要使用其子类,此时如果是向文件中写入内容,所以应该使用FileWriter的子类。

FileWriter类的构造方法定义如下:

publicFileWriter(Filefile)throwsIOException

字符流的操作比字节流操作好在一点,就是可以直接输出字符串了,不用再像之前那样进行转换操作了。

写文件:

1importjava.io.File;
2importjava.io.FileWriter;
3importjava.io.IOException;
4importjava.io.Writer;
5
6publicclassTest16{
7publicstaticvoidmain(String[]args)throwsIOException{
8Filef=newFile("d:"+File.separator+"test.txt");
9Writerout=newFileWriter(f);
10Stringstr="HelloWorld";
11out.write(str);
12out.close();
13}
14}

在默认情况下再次输出会覆盖,追加的方法也是在构造函数上加上追加标记

1importjava.io.File;
2importjava.io.FileWriter;
3importjava.io.IOException;
4importjava.io.Writer;
5
6publicclassTest17{
7publicstaticvoidmain(String[]args)throwsIOException{
8Filef=newFile("d:"+File.separator+"test.txt");
9Writerout=newFileWriter(f,true);//追加
10Stringstr="\r\nHelloWorld";
11out.write(str);
12out.close();
13}
14}


字符输入流:Reader

Reader是使用字符的方式从文件中取出数据,Reader类的定义如下:

publicabstractclassReaderextendsObjectsimplementsReadableCloseable

Reader本身也是抽象类,如果现在要从文件中读取内容,则可以直接使用FileReader子类。

FileReader的构造方法定义如下:

publicFileReader(Filefile)throwsFileNotFoundException

以字符数组的形式读取出数据:

1importjava.io.File;
2importjava.io.FileReader;
3importjava.io.IOException;
4importjava.io.Reader;
5
6publicclassTest18{
7publicstaticvoidmain(String[]args)throwsIOException{
8Filef=newFile("d:"+File.separator+"test.txt");
9Readerinput=newFileReader(f);
10char[]c=newchar[1024];
11intlen=input.read(c);
12input.close();
13System.out.println(newString(c,0,len));
14}
15}

也可以用循环方式,判断是否读到底:

1importjava.io.File;
2importjava.io.FileReader;
3importjava.io.IOException;
4importjava.io.Reader;
5
6publicclassTest19{
7publicstaticvoidmain(String[]args)throwsIOException{
8Filef=newFile("d:"+File.separator+"test.txt");
9Readerinput=newFileReader(f);
10char[]c=newchar[1024];
11inttemp=0;
12intlen=0;
13while((temp=input.read())!=-1){
14c[len]=(char)temp;
15len++;
16}
17input.close();
18System.out.println(newString(c,0,len));
19}
20}

字节流与字符流的区别

字节流和字符流使用是非常相似的,那么除了操作代码的不同之外,还有哪些不同呢?

字节流在操作的时候本身是不会用到缓冲区(内存)的,是与文件本身直接操作的,而字符流在操作的时候是使用到缓冲区的

字节流在操作文件时,即使不关闭资源(close方法),文件也能输出,但是如果字符流不使用close方法的话,则不会输出任何内容,说明字符流用的是缓冲区,并且可以使用flush方法强制进行刷新缓冲区,这时才能在不close的情况下输出内容

那开发中究竟用字节流好还是用字符流好呢?

在所有的硬盘上保存文件或进行传输的时候都是以字节的方法进行的,包括图片也是按字节完成,而字符是只有在内存中才会形成的,所以使用字节的操作是最多的。

如果要java程序实现一个拷贝功能,应该选用字节流进行操作(可能拷贝的是图片),并且采用边读边写的方式(节省内存)。

分享到:
评论

相关推荐

    java 字节流和字符流的区别详解

    Java 字节流和字符流的区别详解 Java 中的字节流和字符流是两种不同的输入 / 输出流,它们之间的区别是非常重要的。在本文中,我们将详细介绍 Java 中的字节流和字符流的区别,并通过示例代码来演示它们的使用。 ...

    io流详解,字符流和字节流代码

    本文件包“io流详解,字符流和字节流代码”显然是针对Java IO流的深入学习资源,包含了代码示例和可能的可视化解释。 IO流分为两大类:字节流(Byte Stream)和字符流(Character Stream)。字节流处理的是8位的数据...

    详解Java中字符流与字节流的区别

    Java中字符流与字节流的区别 Java中的流是对字节序列的抽象,可以想象有一个水管,不再是水,而是字节序列。流具有一个“流动的方向”,从中读入一个字节序列的对象被称为输入流;能够向其写入一个字节序列的对象被...

    Java字符流和字节流

    ### Java字符流和字节流详解 #### 一、引言 在Java中,I/O流主要用于处理输入输出操作,包括文件读写等。根据处理数据类型的不同,Java I/O流主要分为字节流和字符流两大类。字节流处理的是8位的字节数据,而字符流...

    Java IO字符流和字节流

    #### 三、字节流与字符流的选择 选择使用字节流还是字符流取决于具体的应用场景: - 如果需要处理的是文本数据,那么通常使用字符流更为方便,因为它可以直接处理字符,避免了字符编码转换的问题。 - 如果需要处理...

    详解JAVA 字节流和字符流

    Java字节流和字符流是Java I/O流处理中非常基础且重要的概念,它们是用于文件读写操作的两大类型。Java字节流主要包括InputStream和OutputStream两个基类及其子类,字符流则包括Reader和Writer两个基类及其子类。 ...

    day09_字节流、字符流-答案1

    【Java字节流与字符流】 在Java中,数据的输入输出主要通过流(Stream)来完成。字节流(Byte Stream)处理的是原始的字节数据,适合处理任何类型的数据,包括图像、音频、视频等。字符流(Character Stream)则是...

    java字节与字符

    #### 三、字符流详解 - **字符输出流**:用于将字符数据写出到文件或设备中,顶级抽象类为`Writer`。 - **字符输入流**:用于从文件或设备中读取字符数据,顶级抽象类为`Reader`。 ##### 3.1 Writer `Writer`是...

    32.9、java io详解1

    Java IO 之字节流和字符流的区别 Java 中的 IO 操作是指对文件、网络或其他输入/输出设备的读写操作。Java 中的 IO 操作可以分为两大类:字节流和字符流。字节流和字符流是 Java 中的两种基本的 IO 操作方式,它们...

    io流详解代码

    - 根据处理的数据类型,流分为字节流(Byte Stream)和字符流(Character Stream)。字节流处理单个字节的数据,如图片、音频等;字符流处理Unicode字符,适合文本数据。 - 根据操作方式,流又可分为节点流(如...

    java字节字符转换流操作详解

    字节流处理的是字节序列,而字符流处理的是字符序列,这两者在处理文本文件时尤其重要,因为不同的编码方式(如UTF-8、GBK等)会导致字节序列与字符序列之间的转换有所不同。 1. **基本概念** - **文本和文本文件*...

    java中的io流详解

    Java 中的 IO 流详解 Java 中的 IO 流是指在 Java 语言中对输入输出操作的抽象,主要用于实现程序与外部设备和其他计算机之间的交流,尤其是对磁盘文件的操作。IO 流的概念是在 1984 年由 C 语言第一次引入的,后来...

    Java_输入输出流及文件读写详解.docx

    从 I/O 类的体系结构开始,讲解了 Java 中的输入输出流的分类,包括字节流和字符流,以及它们之间的区别。然后,着重介绍了字节输入流的父类 InputStream,包括其方法和使用场景。 在 I/O 类的体系结构中,Java 中...

    java IO 字节流详解及实例代码

    Java IO 字节流详解及实例代码 Java IO 字节流是 Java 编程语言中最基本的输入/输出机制。它提供了各种输入/输出流,以满足不同的输入/输出需求。在本文中,我们将详细介绍 Java IO 字节流的概念、分类、实例代码和...

    Java输入输出流及文件读写详解

    #### 字节流与字符流 在Java中,根据数据序列的基本单位不同,流又分为字节流和字符流两种: - **字节流**: 数据序列以`byte`为单位,通常用于处理二进制数据或非文本数据。 - **字符流**: 数据序列以`char`为单位...

    java IO流操作详解源代码下载

    Java IO库中的流分为字节流和字符流两大类,每类又分为输入流和输出流。字节流处理单个字节的数据,包括InputStream和OutputStream家族;字符流处理Unicode字符,包括Reader和Writer家族。此外,还有双向流...

    2024IO流-字符流-HM

    字符流主要用于处理文本数据,它能够避免字节流读取中文等多字节字符时可能出现的乱码问题。字符流是按照字符为单位进行读写的,每个字符通常占用两个字节。Java提供了一系列的字符流类,如`FileReader`和`...

    Java字符流.pdf

    ### Java字符流详解 #### 一、概述 Java中的字符流是一种高级的输入输出机制,主要用于处理文本数据。与字节流不同,字符流处理的基本单位是字符而非字节,因此更加适合处理文本数据。字符流是基于Unicode标准设计...

Global site tag (gtag.js) - Google Analytics