Java - 字节流 & 字符流区别

简介: Java - 字节流 & 字符流区别

字节流与和字符流的使用非常相似,两者除了操作代码上的不同之外,是否还有其他的不同呢?

实际上字节流在操作时本身不会用到缓冲区(内存),是文件本身直接操作的,而字符流在操作时使用了缓冲区,通过缓冲区再操作文件,如图所示。

image.png

下面以两个写文件的操作为主进行比较,但是在操作时字节流和字符流的操作完成之后都不关闭输出流

/* 范例:使用字节流不关闭执行 */importjava.io.File;
importjava.io.FileOutputStream;
importjava.io.OutputStream;
publicclassDemo1 {
publicstaticvoidmain(String[] args) throwsException { // 异常抛出,不处理// 第1步:使用File类找到一个文件Filef=newFile("d:"+File.separator+"test.txt"); // 声明File对象// 第2步:通过子类实例化父类对象OutputStreamout=null; // 准备好一个输出的对象out=newFileOutputStream(f); // 通过对象多态性进行实例化// 第3步:进行写操作Stringstr="Hello World!!!"; // 准备一个字符串byteb[] =str.getBytes(); // 字符串转byte数组out.write(b); // 将内容输出// 第4步:关闭输出流// out.close(); // 此时没有关闭    }
}

程序运行结果:

image.png

此时没有关闭字节流操作,但是文件中也依然存在了输出的内容,证明字节流是直接操作文件本身的。而下面继续使用字符流完成,再观察效果。

/* 范例:使用字符流不关闭执行 */importjava.io.File;
importjava.io.FileWriter;
importjava.io.Writer;
publicclassDemo2 {
publicstaticvoidmain(String[] args) throwsException { // 异常抛出, 不处理// 第1步:使用File类找到一个文件Filef=newFile("d:"+File.separator+"test.txt"); // 声明File对象// 第2步:通过子类实例化父类对象Writerout=null; // 准备好一个输出的对象out=newFileWriter(f); // 通过对象多态性进行实例化// 第3步:进行写操作Stringstr="Hello World!!!"; // 准备一个字符串out.write(str); // 将内容输出// 第4步:关闭输出流// out.close(); // 此时没有关闭    }
}

程序运行结果:

image.png

程序运行后会发现文件中没有任何内容,这是因为字符流操作时使用了缓冲区,而在关闭字符流时会强制性地将缓冲区中的内容进行输出,但是如果程序没有关闭,则缓冲区中的内容是无法输出的,所以得出结论:字符流使用了缓冲区,而字节流没有使用缓冲区。

提问:什么叫缓冲区?在很多地方都碰到缓冲区这个名词,那么到底什么是缓冲区?又有什么作用呢?

回答:缓冲区可以简单地理解为一段内存区域。可以简单地把缓冲区理解为一段特殊的内存。

某些情况下,如果一个程序频繁地操作一个资源(如文件或数据库),则性能会很低,此时为了提升性能,就可以将一部分数据暂时读入到内存的一块区域之中,以后直接从此区域中读取数据即可,因为读取内存速度会比较快,这样可以提升程序的性能。

在字符流的操作中,所有的字符都是在内存中形成的,在输出前会将所有的内容暂时保存在内存之中,所以使用了缓冲区暂存数据。如果想在不关闭时也可以将字符流的内容全部输出,则可以使用 Writer类 中的 flush() 方法完成。

/* 范例:强制性清空缓冲区 */importjava.io.File;
importjava.io.FileWriter;
importjava.io.Writer;
publicclassDemo3 {
publicstaticvoidmain(String[] args) throwsException { // 异常抛出不处理// 第1步:使用File类找到一个文件Filef=newFile("d:"+File.separator+"test.txt");// 声明File对象// 第2步:通过子类实例化父类对象Writerout=null; // 准备好一个输出的对象out=newFileWriter(f); // 通过对象多态性进行实例化// 第3步:进行写操作Stringstr="Hello World!!!"; // 准备一个字符串out.write(str); // 将内容输出out.flush(); // 强制性清空缓冲区中的内容// 第4步:关闭输出流// out.close(); // 此时没有关闭    }
}

程序运行结果:

image.png

此时,文件中已经存在了内容,更进一步证明内容是保存在缓冲区的。这一点在读者日后的开发中要特别引起注意。

提问:使用字节流好还是字符流好?学习完字节流和字符流的基本操作后,已经大概地明白了操作流程的各个区别,那么在开发中是使用字节流好还是字符流好呢?

回答:使用字节流更好。在回答之前,先为读者讲解这样的一个概念,所有的文件在硬盘或在传输时都是以字节的方式进行的,包括图片等都是按字节的方式存储的,而字符是只有在内存中才会形成,所以在开发中,字节流使用较为广泛。字节流与字符流主要的区别是他们的的处理方式。

 

流分类:

  1. Java的字节流:InputStream是所有字节输入流的祖先,而OutputStream是所有字节输出流的祖先。
  2. Java的字符流:Reader是所有读取字符串输入流的祖先,而Writer是所有输出字符串的祖先。
  • Ps:InputStream,OutputStream,Reader,Writer都是抽象类。所以不能直接 new。


应用场景:

  1. 二进制格式(只要不能确定是纯文本的):InputStream,OutputStream及其子类(字节流)。
  2. 纯文本格式(含纯英文与汉字或其他编码方式):Reader,Writer及其子类(字符流)。



最后:

字节流是最基本的,所有的 InputStream 和 OutputStream 的子类都是,主要用在处理二进制数据,它是按字节来处理的,但实际中很多的数据是文本,又提出了字符流的概念,它是按虚拟机的encode来处理,也就是要进行字符集的转化。这两个之间通过 InputStreamReader,OutputStreamWriter 来关联,实际上是通过 byte[] 和 String 来关联。在实际开发中出现的汉字问题实际上都是在字符流和字节流之间转化不统一而造成的。在从字节流转化为字符流时,实际上就是byte[]转化为String时,public String(byte bytes[], String charsetName) 有一个关键的参数字符集编码,通常我们都省略了,那系统就用操作系统的lang,而在字符流转化为字节流时,实际上是String转化为byte[]时,byte[]、String.getBytes(String charsetName) 也是一样的道理。

目录
相关文章
|
6月前
|
安全 Java API
Java SE 与 Java EE 区别解析及应用场景对比
在Java编程世界中,Java SE(Java Standard Edition)和Java EE(Java Enterprise Edition)是两个重要的平台版本,它们各自有着独特的定位和应用场景。理解它们之间的差异,对于开发者选择合适的技术栈进行项目开发至关重要。
1024 1
|
7月前
|
Java 测试技术
Java浮点类型详解:使用与区别
Java中的浮点类型主要包括float和double,它们在内存占用、精度范围和使用场景上有显著差异。float占用4字节,提供约6-7位有效数字;double占用8字节,提供约15-16位有效数字。float适合内存敏感或精度要求不高的场景,而double精度更高,是Java默认的浮点类型,推荐在大多数情况下使用。两者都存在精度限制,不能用于需要精确计算的金融领域。比较浮点数时应使用误差范围或BigDecimal类。科学计算和工程计算通常使用double,而金融计算应使用BigDecimal。
2661 102
|
8月前
|
存储 缓存 人工智能
Java int和Integer的区别
本文介绍了Java中int与Integer的区别及==与equals的比较机制。Integer是int的包装类,支持null值。使用==比较时,int直接比较数值,而Integer比较对象地址;在-128至127范围内的Integer值可缓存,超出该范围或使用new创建时则返回不同对象。equals方法则始终比较实际数值。
272 0
|
8月前
|
安全 算法 Java
Java 中 synchronized 与 AtomicInteger 的区别
在Java多线程编程中,`synchronized`和`AtomicInteger`均用于实现线程安全,但原理与适用场景不同。`synchronized`是基于对象锁的同步机制,适用于复杂逻辑和多变量同步,如银行转账;而`AtomicInteger`采用CAS算法,适合单一变量的原子操作,例如计数器更新。二者各有优劣,应根据具体需求选择使用。
225 0
|
9月前
|
算法 Java 数据库连接
Java 与 C++ 区别深入剖析及应用实例详解
本文深入剖析了Java和C++两种编程语言的区别,从编译与执行机制、面向对象特性、数据类型与变量、内存管理、异常处理等方面进行对比,并结合游戏开发、企业级应用开发、操作系统与嵌入式开发等实际场景分析其特点。Java以跨平台性强、自动内存管理著称,适合企业级应用;C++则因高性能和对硬件的直接访问能力,在游戏引擎和嵌入式系统中占据优势。开发者可根据项目需求选择合适语言,提升开发效率与软件质量。附面试资料链接:[点此获取](https://pan.quark.cn/s/4459235fee85)。
762 0
|
9月前
|
存储 Java C语言
Java List 复制:浅拷贝与深拷贝方法及区别
我是小假 期待与你的下一次相遇 ~
857 1
|
10月前
|
Java
Java 中 Exception 和 Error 的区别
在 Java 中,`Exception` 和 `Error` 都是 `Throwable` 的子类,用于表示程序运行时的异常情况。`Exception` 表示可被捕获和处理的异常,分为受检异常(Checked)和非受检异常(Unchecked),通常用于程序级别的错误处理。而 `Error` 表示严重的系统级问题,如内存不足或 JVM 错误,一般不建议捕获和处理。编写程序时应重点关注 `Exception` 的处理,确保程序稳定性。
306 0
|
5月前
|
JSON 网络协议 安全
【Java】(10)进程与线程的关系、Tread类;讲解基本线程安全、网络编程内容;JSON序列化与反序列化
几乎所有的操作系统都支持进程的概念,进程是处于运行过程中的程序,并且具有一定的独立功能,进程是系统进行资源分配和调度的一个独立单位一般而言,进程包含如下三个特征。独立性动态性并发性。
269 1
|
5月前
|
JSON 网络协议 安全
【Java基础】(1)进程与线程的关系、Tread类;讲解基本线程安全、网络编程内容;JSON序列化与反序列化
几乎所有的操作系统都支持进程的概念,进程是处于运行过程中的程序,并且具有一定的独立功能,进程是系统进行资源分配和调度的一个独立单位一般而言,进程包含如下三个特征。独立性动态性并发性。
285 1