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) 也是一样的道理。

目录
相关文章
|
2月前
|
Java 程序员
Java社招面试题:& 和 && 的区别,HR的套路险些让我翻车!
小米,29岁程序员,分享了一次面试经历,详细解析了Java中&和&&的区别及应用场景,展示了扎实的基础知识和良好的应变能力,最终成功获得Offer。
83 14
|
28天前
|
Java
java中面向过程和面向对象区别?
java中面向过程和面向对象区别?
25 1
|
2月前
|
存储 缓存 安全
java 中操作字符串都有哪些类,它们之间有什么区别
Java中操作字符串的类主要有String、StringBuilder和StringBuffer。String是不可变的,每次操作都会生成新对象;StringBuilder和StringBuffer都是可变的,但StringBuilder是非线程安全的,而StringBuffer是线程安全的,因此性能略低。
60 8
|
2月前
|
Java
Java代码解释++i和i++的五个主要区别
本文介绍了前缀递增(++i)和后缀递增(i++)的区别。两者在独立语句中无差异,但在赋值表达式中,i++ 返回原值,++i 返回新值;在复杂表达式中计算顺序不同;在循环中虽结果相同但使用方式有别。最后通过 `Counter` 类模拟了两者的内部实现原理。
Java代码解释++i和i++的五个主要区别
|
3月前
|
Java
通过Java代码解释成员变量(实例变量)和局部变量的区别
本文通过一个Java示例,详细解释了成员变量(实例变量)和局部变量的区别。成员变量属于类的一部分,每个对象有独立的副本;局部变量则在方法或代码块内部声明,作用范围仅限于此。示例代码展示了如何在类中声明和使用这两种变量。
|
3月前
|
Java
Java代码解释静态代理和动态代理的区别
### 静态代理与动态代理简介 **静态代理**:代理类在编译时已确定,目标对象和代理对象都实现同一接口。代理类包含对目标对象的引用,并在调用方法时添加额外操作。 **动态代理**:利用Java反射机制在运行时生成代理类,更加灵活。通过`Proxy`类和`InvocationHandler`接口实现,无需提前知道接口的具体实现细节。 示例代码展示了两种代理方式的实现,静态代理需要手动创建代理对象,而动态代理通过反射机制自动创建。
|
3月前
|
缓存 算法 Java
Java 中线程和纤程Fiber的区别是什么?
【10月更文挑战第14天】
104 0
|
3月前
|
自然语言处理 Java 数据处理
Java IO流全解析:字节流和字符流的区别与联系!
Java IO流全解析:字节流和字符流的区别与联系!
115 1
|
7月前
|
自然语言处理 Java 数据处理
Java IO流全解析:字节流和字符流的区别与联系!
【6月更文挑战第26天】Java IO流涵盖字节流与字符流。字节流(InputStream/OutputStream)处理数据单位为字节,适用于二进制和文本,而字符流(Reader/Writer)专注于文本,处理单位为字符,处理编码转换。字符流在字节流基础上添加编码处理,以装饰器模式实现。文件复制示例展示了两者区别:字节流直接复制所有数据,字符流处理字符编码。理解并选择适当流类型对优化程序至关重要。
156 0
|
8月前
|
存储 缓存 Java
【Java IO系列】那字节流和字符流有什么区别?
而如果使用缓存流,一次性从文件里读取多个字节到缓存中,可以减少系统调用同时也减少了磁盘读取,提高了读取的效率。所以字符流是一个很方便的流了,没有必要把一个方便的流转换成一个不方便的流。,涉及到用户空间和内核空间之间的上下文切换,这些切换是很昂贵的。从输入流读取下一个数据字节,值字节以0到255范围内的。好的面试官,Java IO有两个参与对象,一个是。,这个基类提供了3个方法可以来读取字节流。,同样是提供了3个方法来支持字符流读取。好的,有这些不同之处,主要是3个方面。是这样的,虽然字节流比字符流的。
107 1
【Java IO系列】那字节流和字符流有什么区别?