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

article/2025/9/16 12:16:11

原文地址:https://www.cnblogs.com/DONGb/p/7844123.html


1. 流的概念

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

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

2. 字节流与字符流

在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来表示的值就得用字符流来读取!比如说汉字.

3. 流读写操作流程

操作流程:

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

a 使用File类打开一个文件

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

c 进行读/写操作

d 关闭输入/输出

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

4. 字节流具体使用

字节流主要是操作byte类型数据,以byte数组为准,主要操作类就是OutputStream、InputStream。

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

public abstract class OutputStream extends Object implements Closeable,Flushable

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

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

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

FileOutputStream类的构造方法如下:

public FileOutputStream(File file)throws FileNotFoundException

写数据实例:

import java.io.File;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.OutputStream;public class InputStreamStudy {public static void main(String[] args) throws IOException {File f = new File("d:" + File.separator+"test.txt");OutputStream out=new FileOutputStream(f);//如果文件不存在会自动创建String str="Hello World";byte[] b=str.getBytes();out.write(b);//因为是字节流,所以要转化成字节数组进行输出out.close();}
}

测试截图:

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

import java.io.File;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.OutputStream;public class InputStreamStudy {public static void main(String[] args) throws IOException {File f = new File("d:" + File.separator + "test.txt");OutputStream out = new FileOutputStream(f);//如果文件不存在会自动创建String str = "Hello World";byte[] b = str.getBytes();for (int i = 0; i < b.length; i++) {out.write(b[i]);}out.close();}
}

测试截图:

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

public FileOutputStream(File file,boolean append)throws FileNotFoundException

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

import java.io.File;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.OutputStream;public class InputStreamStudy {public static void main(String[] args) throws IOException {File f = new File("d:" + File.separator + "test.txt");OutputStream out = new FileOutputStream(f, true);//追加内容String str = "\r\nHello World";byte[] b = str.getBytes();for (int i = 0; i < b.length; i++) {out.write(b[i]);}out.close();}
}

运行截图:

文件中换行为:\r\n

字节输入流:InputStream

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

public abstract class InputStream extends Object implements Closeable

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

观察FileInputStream类的构造方法:

public FileInputStream(File file)throws FileNotFoundException

读文件实例:

import java.io.*;public class InputStreamStudy {public static void main(String[] args) throws IOException {File f = new File("d:" + File.separator+"test.txt");InputStream in=new FileInputStream(f);byte[] b=new byte[1024];int len=in.read(b);in.close();System.out.println(new String(b,0,len));}
}

运行截图:

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

import java.io.*;public class InputStreamStudy {public static void main(String[] args) throws IOException {File f = new File("d:" + File.separator+"test.txt");InputStream in=new FileInputStream(f);System.out.println(f.length());byte[] b=new byte[(int) f.length()];in.read(b);in.close();System.out.println(new String(b));}
}

运行截图:

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

import java.io.*;public class InputStreamStudy {public static void main(String[] args) throws IOException {File f = new File("d:" + File.separator+"test.txt");InputStream in=new FileInputStream(f);byte[] b=new byte[(int) f.length()];for(int i=0;i<b.length;i++){b[i]=(byte) in.read();}in.close();System.out.println(new String(b));}
}

运行截图:

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

import java.io.*;public class InputStreamStudy {public static void main(String[] args) throws IOException {File f = new File("d:" + File.separator+"test.txt");InputStream in=new FileInputStream(f);byte[] b=new byte[1024];int temp=0;int len=0;while((temp=in.read())!=-1){//-1为文件读完的标志b[len]=(byte) temp;len++;}in.close();System.out.println(new String(b,0,len));}
}

运行截图:

5. 字符流具体使用

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

字符输出流:Writer。

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

public abstract class Writer extends Object implements Appendable,Closeable,Flushable

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

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

public FileWriter(File file)throws IOException

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

写文件:

import java.io.*;public class InputStreamStudy {public static void main(String[] args) throws IOException {File f = new File("d:" + File.separator + "test.txt");Writer out = new FileWriter(f);String str = "Hello 字符流";out.write(str);out.close();}
}

运行截图:

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

import java.io.*;public class InputStreamStudy {public static void main(String[] args) throws IOException {File f = new File("d:" + File.separator+"test.txt");Writer out=new FileWriter(f,true);//追加String str="\r\nHello 字符流";out.write(str);out.close();}
}

运行截图:

字符输入流:Reader

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

public abstract class Reader extends Objects implements Readable,Closeable

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

FileReader的构造方法定义如下:

public FileReader(File file)throws FileNotFoundException

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

import java.io.*;public class InputStreamStudy {public static void main(String[] args) throws IOException {File f = new File("d:" + File.separator+"test.txt");Reader input=new FileReader(f);char[] c=new char[1024];int len=input.read(c);input.close();System.out.println(new String(c,0,len));}
}

运行截图:

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

import java.io.*;public class InputStreamStudy {public static void main(String[] args) throws IOException {File f = new File("d:" + File.separator+"test.txt");Reader input=new FileReader(f);char[] c=new char[1024];int temp=0;int len=0;while((temp=input.read())!=-1){c[len]=(char) temp;len++;}input.close();System.out.println(new String(c,0,len));}
}

运行截图:

6. 总结

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

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

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

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

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

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

补充

测试使用不同编码格式使用字节流写入文件,读出:

import java.io.*;public class Main {public static void main(String[] args) throws IOException {File f = new File("/Users/sh/desktop/" + "test.txt");//如果文件不存在会自动创建OutputStream out = new FileOutputStream(f);String str = "你好";byte[] b = str.getBytes();//因为是字节流,所以要转化成字节数组进行输出out.write(b);out.close();File f1 = new File("/Users/sh/desktop/" + "test.txt");InputStream in = new FileInputStream(f1);byte[] b1 = new byte[1024];int temp = 0;int len = 0;//-1为文件读完的标志while ((temp = in.read()) != -1) {b1[len] = (byte) temp;len++;}in.close();System.out.println(new String(b1, 0, len));}
}

测试结果如下:

 使用相同的编码,byte[] b = str.getBytes("GBK"); 修改为 byte[] b = str.getBytes(); 

测试结果如下:

 


http://chatgpt.dhexx.cn/article/OA8hj6Bs.shtml

相关文章

字节流和字符流详解

1.流的概念 在编程中是一种抽象的概念&#xff0c;就好比“水流”&#xff0c;从一段流向另一端在程序中所有的数据都是以流的方式进行传输或保存的&#xff0c;程序需要数据的时候要使用输入流读取数据&#xff0c;而当程序需要将一些数据保存起来的时候&#xff0c;就要使用…

字符流与字节流的区别

字符流与字节流的区别 &#xff08;尊重劳动成果&#xff0c;转载请注明出处&#xff1a;https://blog.csdn.net/cynhafa/article/details/6882061 ) 字节流与和字符流的使用非常相似&#xff0c;两者除了操作代码上的不同之外&#xff0c;是否还有其他的不同呢&#xff1f; …

java 字节流与字符流的区别

字节流与和字符流的使用非常相似&#xff0c;两者除了操作代码上的不同之外&#xff0c;是否还有其他的不同呢&#xff1f; 实际上字节流在操作时本身不会用到缓冲区&#xff08;内存&#xff09;&#xff0c;是文件本身直接操作的&#xff0c;而字符流在操作时使用了缓冲区&a…

字符流和字节流的区别

字符流和字节流的区别 字符流和字节流的区别&#xff1a; 1、读写单位不同&#xff1a;字节流以字节&#xff08;8bit&#xff09;为单位&#xff0c;字符流以字符为单位&#xff0c;根据码表映射字符&#xff0c;一次可能读多个字节。 2、处理对象不同&#xff1a;字节流能处…

IO流总结——字节流与字符流的区别

字节流与字符流的区别&#xff1a; 字节流与字符流的区别&#xff1a; 所有文件的储存是都是字节&#xff08;byte&#xff09;的储存&#xff0c;在磁盘上保留的并不是文件的字符而是先把字符编码成字节&#xff0c;再储存这些字节到磁盘。在读取文件&#xff08;特别是文本文…

6.移动端布局-rem布局

1.rem基础 优点&#xff1a;可以通过修改html里边文字的大小来改变页面中其他元素的大小&#xff0c;可以实现整体控制 1.1 rem单位 rem(root em)是一个相对单位&#xff0c;类似于em。 em是相对于自身元素字体大小&#xff08;若自身没有设置font-size则基础父元素的字体大…

移动端开发——rem布局

目录 前言 一、rem布局 二、rem的基础知识 三、媒体查询 1.媒体查询的使用语法 &#xff08;一&#xff09;mediatype 媒体类型 &#xff08;二&#xff09;关键字 &#xff08;三&#xff09;媒体特性 2.媒体查询和rem组合 3.引入样式 四、适配方案 1.技术方案1 …

Vue 使用 rem布局

Vue 使用 rem布局 1、rem布局需要安装两个插件 npm i lib-flexible -S npm i postcss-pxtorem5.1.1 -D 最新版本postcss-pxtorem需要POSTSS 8&#xff0c;安装老版本可用 lib-flexible: 根据移动端屏幕大小而对应改变html的fontSize postcss-pxtorem&#xff1a; 根据html的fo…

移动端rem布局基本介绍及原理

rem布局 em和rem的认识 在布局中&#xff0c;除了px之外&#xff0c;还有两个常见的单位&#xff0c;em和rem em&#xff1a; 相对于当前元素的字体大小→ 1em 当前标签的font-size rem&#xff1a; 相对于根元素&#xff08;html&#xff09;的字体大小→ 1rem html标签的…

JavaScript-rem布局

JavaScript-rem布局 一、什么是Rem rem和em很容易混淆&#xff0c;其实两个都是css的单位&#xff0c;并且也都是相对单位&#xff0c;现有的em&#xff0c;css3才引入的rem&#xff0c;在介绍rem之前&#xff0c;我们先来了解下em: em作为font-size的单位时&#xff0c;其代…

Rem布局的原理解析

Rem布局的原理解析 tobAlier关注 22017.10.25 11:27:32字数 2,630阅读 12,367 什么是Rem rem和em很容易混淆&#xff0c;其实两个都是css的单位&#xff0c;并且也都是相对单位&#xff0c;现有的em&#xff0c;css3才引入的rem&#xff0c;在介绍rem之前&#xff0c;我们先…

html5如何利用rem实现自适应布局,使用Rem布局实现自适应

之前写过一篇移动端适配的文章,很长,内容太多,看得容易凌乱,重新写个通熟易懂版的。 为什么要自适应? 比如,对于一个移动端页面,设计师给的视觉稿画布宽 750,视觉稿中的一个黄色区块的尺寸是 702 x 300,并在画板中居中。我们希望在任何一个设备中的呈现比例都与视觉稿…

Rem布局

Rem布局?什么东西???Rem布局又称等比缩放布局,是移动端布局中常用的一种布局方式。Rem布局可以通过两种方式实现,一种通过JavaScript实现;还有一种通过vw实现。在这里小编主要讲述第二种方式。 下面小编先简单的说一下rem布局的优缺点: 优点:在不同的设备下看起来比较…

(精中求精) rem适配布局

1.适配导读&#xff1a; 什么是适配布局&#xff1f;与flex或者流式布局又有什么区别&#xff1f; 所谓的适配布局&#xff0c;是让页面盒子的高度&#xff0c;宽度&#xff0c;内外边距&#xff0c;边框大小&#xff0c;文字的大小&#xff0c;定位的元素位置等能够根据屏幕…

移动端布局(三) rem布局及原理

什么是rem 首先来了解一下什么是em: 作为font-size的单位时,其代表父元素的字体大小,em作为其他属性单位时,代表自身字体大小。 s1、s2、s5、s6的font-size和line-height分别是多少px? <div class"p1"><div class"s1">1</div><di…

安装Mysql

设置密码 检测安装是否成功 切换到安装目录的bin下C:\Program Files\MySQL\MySQL Server 8.0\bin&#xff0c;按住shift&#xff0c;单击鼠标右键&#xff0c;点选在此处代开命令窗口 在命令行中输入mysql -u root -p,之后输入之前设定的密码如果出现如下图所示&#xff0c;…

ODBC + WIN32 API 访问MYSQL 数据库实现简单QQ用户注册和登录

//在IT行业&#xff0c;不懂数据库是不行的&#xff0c;我自己在学校里还没有学过数据库有关的知识&#xff0c;不过由于前些日子写项目的要求&#xff0c;不得不去琢磨&#xff0c;虽然有很多数据库类别&#xff0c;包括微软的 SQL server &#xff0c;access,ORACLE公司的ORA…

QQ交谈、QQ在线状态代码生成…

原文地址&#xff1a;QQ交谈、QQ在线状态代码生成&#xff0c;非QQ商家可以正常生成 作者&#xff1a;HHeOnline 非QQ商家用户&#xff0c;直接在QQ互联可以生成QQ在线状态&#xff0c;也就是QQ交谈的代码。 直接放到对应的代码里就可以正常使用了。 QQ代码生成地址&#xff1a…

qq群关系数据库 mysql_QQ群关系数据库24.52G mdf源文件下载 附上使用教程

HQY 腾讯 7000W QQ群关系数据库泄漏共24.52G。 QQ24.5gb腾讯QQ群关系数据库,不包含密码。只包含个人名字(群备注名字等信息)。 如果想在线查询更多qq群数据信息可以查看: 一、下载完24.5gQQ数据库后,解压文件 下载后不要把文件改名,打开7z,点击其中一个文件,选择合并文件…

MySQL 数据类型

数值类型 MySQL支持所有标准SQL数值数据类型。 这些类型包括严格数值数据类型(INTEGER、SMALLINT、DECIMAL和NUMERIC)&#xff0c;以及近似数值数据类型(FLOAT、REAL和DOUBLE PRECISION)。 关键字INT是INTEGER的同义词&#xff0c;关键字DEC是DECIMAL的同义词。 BIT数据类型保存…