当前位置: 首页 > 编程笔记 >

详解JAVA 字节流和字符流

黄宏毅
2023-03-14
本文向大家介绍详解JAVA 字节流和字符流,包括了详解JAVA 字节流和字符流的使用技巧和注意事项,需要的朋友参考一下

1、InputStream 和 Reader

InputStream 和 Reader 是所有输入流的抽象基类,本身并不能创建实例来执行输入,但它们将成为所有输入流的模板,所以它们的方法是所有输入流都可使用的方法。

在 InputStream 里包含如下三个方法。

  • int read():从输入流中读取单个字节,返回所读取的字节数据(字节数据可直接转换为int类型)。
  • int read(byte[] b):从输入流中最多读取 b.length 个字节的数据,并将其存储在字节数组 b 中,返回实际读取的字节数。
  • int read(byte[] b, int off, int len):从输入流中最多读取 len 个字节的数据,并将其存储在数组 b 中,放入数组 b 中时,并不是从数组起点开始,而是从 off 位置开始,返回实际读取的字节数。

在 Reader 里包含如下三个方法。

  • int read():从输入流中读取单个字符,返回所读取的字符数据(字符数据可直接转换为int类型)。
  • int read(char[] cbuf):从输入流中最多读取 cbuf.length 个字符的数据,并将其存储在字符数组 cbuf 中,返回实际读取的字符数。
  • int read(char[] chuf, int off, int len):从输入流中最多读取 len 个字符的数据,并将其存储在字符数组 cbuf 中,放入数组 cbuf 中时,并不是从数组起点开始,而是从 off 位置开始,返回实际读取的字符数。

对比 InputStream 和 Reader 所提供的方法,就不难发现这两个基类的功能基本是一样的。

正如前面提到的,InputStream 和 Reader 都是抽象类,本身不能创建实例,但它们分别有一个用于读取文件的输入流:FileInputStream 和 FileReader,它们都是节点流一一会直接和指定文件关联。

下面程序示范了使用 FileInputStream 来读取自身的效果。

public class FileInputStreamTest {
  public static void main(String[] args) throws IOException {
    // 创建字节输入流
    FileInputStream fis = new FileInputStream("F:\\eclipse-workspace\\demo\\src\\com\\jwen\\demo15_3\\FileInputStreamTest.java");
    // 创建一个长度为1024的“竹筒”
    byte[] bbuf = new byte[1024];
    // 用于保存实际读取的字节数
    int hasRead = 0;
    // 使用循环来重复“取水”过程
    while ((hasRead = fis.read(bbuf)) > 0) {
      // 取出“竹筒”中水滴(字节),将字节数组转换成字符串输入!
      System.out.print(new String(bbuf, 0, hasRead));
    }
    // 关闭文件输入流,放在finally块里更安全
    fis.close();
  }
}

上面程序中的粗体字代码是使用 FileInputStream 循环“取水”的过程,运行上面程序,将会输出上面程序的源代码

注意:上面程序创建了一个长度为1024的字节数组来读取该文件,实际上该Java源文件的长度还不到1024字节,也就是说,程序只需要执行一次 read() 方法即可读取全部内容。但如果创建较小长度的字节数组,程序运行时在输出中文注释时就可能出现乱码一一这是因为本文件保存时采用的是 GBK 编码方式,在这种方式下,每个中文字符占2字节,如果 read() 方法读取时只读到了半个中文字符,这将导致乱码。

上面程序最后使用了 fis.close() 来关闭该文件输入流,与 JDBC 编程一样,程序里打开的文件 IO 资源不属于内存里的资源,垃圾回收机制无法回收该资源,所以应该显式关闭文件资源。Java 7 改写了所有的 IO 资源类,它们都实现了 AutoCloseable 接口,因此都可通过自动关闭资源的 try 语句来关闭这些 IO 流。下面程序使用 FileReader 来读取文件本身。

public class FileReaderTest {
  public static void main(String[] args) {
    try (
      // 创建字符输入流
      FileReader fr = new FileReader("F:\\eclipse-workspace\\demo\\src\\com\\jwen\\demo15_3\\FileReaderTest.java")) {
      // 创建一个长度为32的“竹筒”
      char[] cbuf = new char[32];
      // 用于保存实际读取的字符数
      int hasRead = 0;
      // 使用循环来重复“取水”过程
      while ((hasRead = fr.read(cbuf)) > 0) {
        // 取出“竹筒”中水滴(字符),将字符数组转换成字符串输入!
        System.out.print(new String(cbuf, 0, hasRead));
      }
    } catch (IOException ex) {
      ex.printStackTrace();
    }
  }
}

上面的 FileReaderTest.java 程序与前面的 FileInputStreamTest.java 并没有太大的不同,程序只是将字符数组的长度改为32,这意味着程序需要多次调用 read() 方法才可以完全读取输入流的全部数据。程序最后使用了自动关闭资源的 try 语句来关闭文件输入流,这样可以保证输入流一定会被关闭。

除此之外,InputStream 和 Reader 还支持如下几个方法来移动记录指针。

  • void mark(int readAheadLimit):在记录指针当前位置记录一个标记(mark).
  • boolean markSupported():判断此输入流是否支持 mark() 操作,即是否支持记录标记。
  • void reset():将此流的记录指针重新定位到上一次记录标记(mark)的位置。
  • long skip(long n):记录指针向前移动个字节/字符。

2、OutputStream 和 Writer

OutputStream 和 Writer 也非常相似,两个流都提供了如下三个方法。

  • void write(int c):将指定的字节/字符输出到输出流中,其中 c 既可以代表字节,也可以代表字符。
  • void write(byte[]/char[] buf):将字节数组/字符数组中的数据输出到指定输出流中。
  • void write(byte[]/char[] buf, int off, int len):将字节数组/字符数组中从 off 位置开始,长度为 len 的字节/字符输出到输出流中。

因为字符流直接以字符作为操作单位,所以 Writer 可以用字符串来代替字符数组,即以 String 对象作为参数。Writer 里还包含如下两个方法。

  • void write(String str):将字符串里包含的字符输出到指定输出流中。
  • void write(String str, int off, int len):将字符串里从 off 位置开始,长度为 len 的字符输出到指定输出流中。

下面程序使用 FileInputStream 来执行输入,并使用 FileOutputStream 来执行输出,用以实现复制 FileOutputStreamTest.java 文件的功能。

public class FileOutputStreamTest {
  public static void main(String[] args) {
    try (
      // 创建字节输入流
      FileInputStream fis = new FileInputStream("FileOutputStreamTest.java");
      // 创建字节输出流
      FileOutputStream fos = new FileOutputStream("newFile.txt")) {
      byte[] bbuf = new byte[32];
      int hasRead = 0;
      // 循环从输入流中取出数据
      while ((hasRead = fis.read(bbuf)) > 0) {
        // 每读取一次,即写入文件输出流,读了多少,就写多少。
        fos.write(bbuf, 0, hasRead);
      }
    } catch (IOException ioe) {
      ioe.printStackTrace();
    }
  }
}

运行上面程序,将看到系统当前路径下多了一个文件:newFile.txt,该文件的内容和 FileOutputStreamTest.java 文件的内容完全相同。

注意:使用 Java 的 IO 流执行输出时,不要忘记关闭输出流,关闭输出流除可以保证流的物理资源被回收之外,可能还可以将输出流缓冲区中的数据 flush 到物理节点里(因为在执行 close() 方法之前,自动执行输出流的 flush() 方法)。Java 的很多输出流默认都提供了缓冲功能,其实没有必要刻意去记忆哪些流有缓冲功能、哪些流没有,只要正常关闭所有的输出流即可保证程序正常。

如果希望直接输出字符串内容,则使用 Writer 会有更好的效果,如下程序所示。

public class FileWriterTest {
  public static void main(String[] args) {
    try (FileWriter fw = new FileWriter("poem.txt")) {
      fw.write("锦瑟 - 李商隐\r\n");
      fw.write("锦瑟无端五十弦,一弦一柱思华年。\r\n");
      fw.write("庄生晓梦迷蝴蝶,望帝春心托杜鹃。\r\n");
      fw.write("沧海月明珠有泪,蓝田日暖玉生烟。\r\n");
      fw.write("此情可待成追忆,只是当时已惘然。\r\n");
    } catch (IOException ioe) {
      ioe.printStackTrace();
    }
  }
}

运行上面程序,将会在当前目录下输出一个 poem.txt 文件,文件内容就是程序中输出的内容。

注意:上面程序在输出字符串内容时,字符串内容的最后是\r\n,这是 Windows 平台的换行符,通过这种方式就可以让输出内容换行;如果是 UNIX/Linux/BSD 等平台,则使用 \n 就作为换行符。

以上就是详解JAVA 字节流和字符流的详细内容,更多关于JAVA 字节流和字符流的资料请关注小牛知识库其它相关文章!

 类似资料:
  • 本文向大家介绍java 字节流和字符流的区别详解,包括了java 字节流和字符流的区别详解的使用技巧和注意事项,需要的朋友参考一下 字节流与和字符流的使用非常相似,两者除了操作代码上的不同之外,是否还有其他的不同呢? 实际上字节流在操作时本身不会用到缓冲区(内存),是文件本身直接操作的,而字符流在操作时使用了缓冲区,通过缓冲区再操作文件,如图所示。 下面以两个写文件的操作为主进行比较,但是在操作时

  • 本文向大家介绍详解Java中字符流与字节流的区别,包括了详解Java中字符流与字节流的区别的使用技巧和注意事项,需要的朋友参考一下 本文为大家分析了Java中字符流与字节流的区别,供大家参考,具体内容如下 1. 什么是流     Java中的流是对字节序列的抽象,我们可以想象有一个水管,只不过现在流动在水管中的不再是水,而是字节序列。和水流一样,Java中的流也具有一个“流动的方向”,通常可以从中

  • 问题内容: 请解释什么是字节流和字符流。这些到底是什么意思?Microsoft Word文档是面向字节还是面向字符? 谢谢 问题答案: 流是顺序访问文件的一种方式。字节流逐字节访问文件。字节流适用于任何类型的文件,但不适用于文本文件。例如,如果文件使用unicode编码,并且一个字符用两个字节表示,则字节流将分别处理这些字节,您需要自己进行转换。 字符流将逐字符读取文件。必须为字符流提供文件的编码

  • 我在理解JavaIO类中的一些概念时有些困难。例如,有两种类型的流,字节流和字符流。据我所知,字节流逐个字节读取。 1.如果java中的char存储为16位(2字节)的数据类型,那么我怎么可能使用面向字节的输入流从文件中准确读取char,比如'A',例如FileInputStream? 2。是因为我使用的字符(在ascii图表上大多在0到122之间)存储在分配的两个字节中的一个字节中吗? 3. D

  • 我读了这段代码。正如世外桃源.txt内容使用“测试”。该文件的大小为 4 个字节。如果我使用调试来运行 一次一个字节,每次打开文件后再次打开.txt(用记事本)我依次看到:t--

  • 本文向大家介绍java IO 字节流详解及实例代码,包括了java IO 字节流详解及实例代码的使用技巧和注意事项,需要的朋友参考一下 java IO 字节流详解 1.         如何理解输入输出流? 这是我当初在学习Java IO这一块很难理解的一块,输入输出流我们可必须以一个为参照物:我们以内存为参照物,凡是写入内存的我们叫输入流,从内存中写出的我们叫输出流。看下面的示例图 有了这样的一