百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 编程网 > 正文

Java字节流与字符流解析:概念辨析、核心差异与实际应用示例

yuyutoo 2024-10-14 16:20 6 浏览 0 评论

字节流与字符流是Java中处理输入/输出操作的两种主要方式,各自有特定的使用场景和优势。

基本概念

字节流(Byte Streams)是处理字节数据输入/输出流,是基于字节(Byte)进行数据读写操作的数据流类型。而字符流(Character Streams)则是处理字符数据输入/输出流。字节是计算机中基本的数据单元,一个字节由8位二进制数构成,可以表示256种不同的状态。而字符则是人类用来表示语言符号的抽象概念

使用场景


场景

字节流

处理非文本文件(如图片、音频、视频等)时,应使用字节流。

当需要精确控制数据的读写,或者处理的数据量非常大时,字节流更加合适。

字节流是通用的,可以用于任何类型的数据,包括文本数据。

字符流

当处理文本文件时,应优先考虑使用字符流。

字符流会自动处理字符编码解码的问题,使得读取和写入文本数据更为简单。

字符流提供了更高级别的抽象,可以方便地按行读取文本数据,这在处理文本文件时非常有用。

字节流(Byte Streams)

字节流是处理字节的输入/输出流,主要处理二进制数据。字节流主要由InputStreamOutputStream两个抽象类及其子类组成。字节流是低级别的流,不使用缓冲区,直接对文件进行读写操作。由于字节流不直接处理字符,适用于处理任何类型的数据,包括文本、图片、音频和视频等。

字节流家族

抽象类

子类

子类

InputStream

FileInputStream


FilterInputStream

BufferedInputStream

DataInputStream

PushbackInputStream

ObjectInputStream


PipedInputStream


SequenceInputStream


StringBufferInputStream


ByteArrayInputStream





OutputStream

FileOutputStream


FilterOutputStream

BufferedOutputStream

DataOutputStream

PrintStream

ObjectOutputStream


PipedOutputStream


ByteArrayOutputStream


字节流使用案例

输入/输出流

说明

FileInputStream

从文件读取字节。

FileOutputStream

向文件写入字节。

import java.io.*;
public class ByteStreamExamples {
  /**
  * 使用FileInputStream和FileOutputStream将源文件的内容复制到目标文件。
  *
  * @param srcFilePath 源文件路径。
  * @param destFilePath 目标文件路径。
  * @throws IOException 如果在读取或写入文件时发生I/O错误。
  */
    public static void copyFile(String srcFilePath, String destFilePath) throws IOException {
      try (FileInputStream fis = new FileInputStream(srcFilePath);
      FileOutputStream fos = new FileOutputStream(destFilePath)) {
        byte[] buffer = new byte[4096]; // 缓冲区大小可根据需要调整
        int bytesRead;
        while ((bytesRead = fis.read(buffer)) != -1) {
          fos.write(buffer, 0, bytesRead);
        }
      }
    }
  /**
  * 向指定文件追加字节数组。
  *
  * @param filePath 文件路径,将在该文件末尾追加字节数组。
  * @param bytes 要追加的字节数组。
  * @throws IOException 如果在写入文件时发生I/O错误。
  */
  public static void appendBytesToFile(String filePath, byte[] bytes) throws IOException {
    try (FileOutputStream fos = new FileOutputStream(filePath, true)) { // true表示启用追加模式
      fos.write(bytes);
    }
  }
  public static void main(String[] args) {
    String sourceFile = "source.txt";
    String destinationFile = "destination.txt";
    byte[] dataToAppend = {0x01, 0x02, 0x03, 0x04};
    try {
      ByteStreamExamples.copyFile(sourceFile, destinationFile);
      ByteStreamExamples.appendBytesToFile(destinationFile, dataToAppend);
      System.out.println("文件操作完成成功。");
    } catch (IOException e) {
      System.err.println("在进行文件操作时发生错误:" + e.getMessage());
    }
  }
}

字节流的主要特点

特点

描述

通用性

由于所有数据在计算机内部最终都可以转换为字节序列,字节流具有极高的通用性,几乎可以处理所有类型的数据。

低层次操作

字节流直接对字节进行操作,属于底层I/O操作,对于需要精细控制数据传输过程的情况尤为适用。

无字符编码

字节流本身并不涉及字符编码,对于非文本数据或已知编码格式的文本数据,使用字节流更为高效。

字符流(Character Streams)

字符流是处理字符的输入/输出流,主要处理文本数据。字符流主要由ReaderWriter两个抽象类及其子类组成。字符流会使用缓冲区,通过缓冲区一次读写多个字符,从而提高效率。字符流只能处理文本数据,因为字符流会按照指定的字符集将字符转换为字节进行读写。

字符流家族

抽象类

子类

子类

Reader

BufferedReader

LineNumberReader

InputStreamReader

FileReader

StringReader


PipedReader


CharArrayReader


FilterReader

PushbackReader




Writer

BufferedWriter


OutputStreamWriter

FileWriter

PrintWriter


StringWriter


PipedWriter


CharArrayWriter


FilterWriter


字符流使用案例

输入/输出流

说明

FileReader

从文件读取字符。

FileWriter

向文件写入字符。

import java.io.*;
public class CharacterStreamExamples {
  /**
  * 使用FileReader和FileWriter将源文本文件的内容复制到目标文本文件。
  *
  * @param srcFilePath 源文本文件路径。
  * @param destFilePath 目标文本文件路径。
  * @throws IOException 如果在读取或写入文件时发生I/O错误。
  */
    public static void copyTextFile(String srcFilePath, String destFilePath) throws IOException {
      try (FileReader fr = new FileReader(srcFilePath);
      FileWriter fw = new FileWriter(destFilePath)) {
        char[] buffer = new char[4096]; // 缓冲区大小可根据需要调整
        int charsRead;
        while ((charsRead = fr.read(buffer)) != -1) {
          fw.write(buffer, 0, charsRead);
        }
      }
    }
  /**
  * 向指定文本文件追加字符串。
  *
  * @param filePath 文本文件路径,将在该文件末尾追加字符串。
  * @param text 要追加的字符串。
  * @throws IOException 如果在写入文件时发生I/O错误。
  */
  public static void appendTextToFile(String filePath, String text) throws IOException {
    try (FileWriter fw = new FileWriter(filePath, true)) { // true表示启用追加模式
      fw.write(text);
    }
  }
  public static void main(String[] args) {
    String sourceFile = "source.txt";
    String destinationFile = "destination.txt";
    String textToAppend = "附加的文本内容";
    try {
      CharacterStreamExamples.copyTextFile(sourceFile, destinationFile);
      CharacterStreamExamples.appendTextToFile(destinationFile, textToAppend);
      System.out.println("文件操作完成成功。");
    } catch (IOException e) {
      System.err.println("在进行文件操作时发生错误:" + e.getMessage());
    }
  }
}

字符流的主要特点

特点

描述

面向文本

字符流专为处理文本数据设计,尤其适合处理包含人类可读字符的文件,如纯文本文件、XML文件等。

字符编码透明

字符流在读写过程中自动处理字符编码,程序员无需关心具体的编码细节,只需指定或推断正确的编码方式即可。

高效文本处理

对于大量文本数据,使用字符流能避免逐字节判断字符边界,提高处理效率。

编码问题

在处理文本数据时,字符编码是一个重要的问题。不同的字符编码方式会将字符映射到不同的字节序列。在使用字符流读写文本数据时,需要指定正确的字符编码方式,以避免出现乱码问题。

案例

import java.io.*;
import java.nio.file.Files;
import java.nio.file.Paths;

public class EncodingExample {
  public static void main(String[] args) {
    String inputFile = "input.txt";
    String outputFile = "output.txt";
    String encoding = "UTF-8"; // 设定字符编码为UTF-8

    try {
      // 读取文件,并指定编码方式
      byte[] encoded = Files.readAllBytes(Paths.get(inputFile));
      String content = new String(encoded, encoding); // 将字节数组解码为字符串

      // 处理文本数据(这里仅仅是打印出来)
      System.out.println("读取的内容:" + content);

      // 写入文件,并指定编码方式
      try (BufferedWriter writer = new BufferedWriter(
           new OutputStreamWriter(new FileOutputStream(outputFile), encoding))) {
        writer.write(content); // 将字符串编码为字节序列并写入文件
      }
      System.out.println("文件已成功写入。");

    } catch (IOException e) {
      e.printStackTrace();
    }
  }
}

字节流与字符流的差异

差异

说明

处理单位

字节流以字节为处理单位,而字符流以字符为处理单位。字节是二进制数据的基本单位,适用于任何类型的数据;字符则特指文本数据中的单个符号。

编码处理

字节流不对数据进行编码或解码,直接读写字节序列;字符流则在读写过程中自动进行字符编码和解码,确保字符的正确表示和传输。

适用场景

字节流适用于处理所有类型的数据,特别是二进制文件;字符流则主要应用于处理文本数据,尤其是涉及到字符编码转换的场景。

相关推荐

当 Linux 根分区 (/) 已满时如何释放空间?

根分区(/)是Linux文件系统的核心,包含操作系统核心文件、配置文件、日志文件、缓存和用户数据等。当根分区满载时,系统可能出现无法写入新文件、应用程序崩溃甚至无法启动的情况。常见原因包括:...

玩转 Linux 之:磁盘分区、挂载知多少?

今天来聊聊linux下磁盘分区、挂载的问题,篇幅所限,不会聊的太底层,纯当科普!!1、Linux分区简介1.1主分区vs扩展分区硬盘分区表中最多能存储四个分区,但我们实际使用时一般只分为两...

Linux 文件搜索神器 find 实战详解,建议收藏

在Linux系统使用中,作为一个管理员,我希望能查找系统中所有的大小超过200M文件,查看近7天系统中哪些文件被修改过,找出所有子目录中的可执行文件,这些任务需求...

Linux 操作系统磁盘操作(linux 磁盘命令)

一、文档介绍本文档描述Linux操作系统下多种场景下的磁盘操作情况。二、名词解释...

Win10新版19603推送:一键清理磁盘空间、首次集成Linux文件管理器

继上周四的Build19592后,微软今晨面向快速通道的Insider会员推送Windows10新预览版,操作系统版本号Build19603。除了一些常规修复,本次更新还带了不少新功能,一起来了...

Android 16允许Linux终端使用手机全部存储空间

IT之家4月20日消息,谷歌Pixel手机正朝着成为强大便携式计算设备的目标迈进。2025年3月的更新中,Linux终端应用的推出为这一转变奠定了重要基础。该应用允许兼容的安卓设备...

Linux 系统管理大容量磁盘(2TB+)操作指南

对于容量超过2TB的磁盘,传统MBR分区表的32位寻址机制存在限制(最大支持2.2TB)。需采用GPT(GUIDPartitionTable)分区方案,其支持64位寻址,理论上限为9.4ZB(9....

Linux 服务器上查看磁盘类型的方法

方法1:使用lsblk命令lsblk输出说明:TYPE列显示设备类型,如disk(物理磁盘)、part(分区)、rom(只读存储)等。...

ESXI7虚机上的Ubuntu Linux 22.04 LVM空间扩容操作记录

本人在实际的使用中经常遇到Vmware上安装的Linux虚机的LVM扩容情况,最终实现lv的扩容,大多数情况因为虚机都是有备用或者可停机的情况,一般情况下通过添加一块物理盘再加入vg,然后扩容lv来实...

5.4K Star很容易!Windows读取Linux磁盘格式工具

[开源日记],分享10k+Star的优质开源项目...

Linux 文件系统监控:用脚本自动化磁盘空间管理

在Linux系统中,文件系统监控是一项非常重要的任务,它可以帮助我们及时发现磁盘空间不足的问题,避免因磁盘满而导致的系统服务不可用。通过编写脚本自动化磁盘空间管理,我们可以更加高效地处理这一问题。下面...

Linux磁盘管理LVM实战(linux实验磁盘管理)

LVM(逻辑卷管理器,LogicalVolumeManager)是一种在Linux系统中用于灵活管理磁盘空间的技术,通过将物理磁盘抽象为逻辑卷,实现动态调整存储容量、跨磁盘扩展等功能。本章节...

Linux查看文件大小:`ls`和`du`为何结果不同?一文讲透原理!

Linux查看文件大小:ls和du为何结果不同?一文讲透原理!在Linux运维中,查看文件大小是日常高频操作。但你是否遇到过以下困惑?...

使用 df 命令检查服务器磁盘满了,但用 du 命令发现实际小于磁盘容量

在Linux系统中,管理员或开发者经常会遇到一个令人困惑的问题:使用...

Linux磁盘爆满紧急救援指南:5步清理释放50GB+小白也能轻松搞定

“服务器卡死?网站崩溃?当Linux系统弹出‘Nospaceleft’的红色警报,别慌!本文手把手教你从‘删库到跑路’进阶为‘磁盘清理大师’,5个关键步骤+30条救命命令,快速释放磁盘空间,拯救你...

取消回复欢迎 发表评论: