当前位置:   article > 正文

Java读取文件的N种方法_java 读取文件

java 读取文件

1.概述

在这篇文章里, 我们将探索不同的方式从文件中读取数据

首先, 学习通过标准的的Java类,从classpathURL或者Jar中加载文件。

然后,学习通用BufferedReader, Scanner, StreamTokenizer, DataInputStream, SequenceInputStream, FileChannel读取文件内容。也会讨论如何读取UTF-8编码的文件。

最后,学习Java7和Java8中新的加载和读取文件的技术。

2.准备

2.1 输入文件

这篇文章的很多示例,从名为fileTest.txt的文件读取文本内容,文件包含

Hello,World!

有少量示例, 我们会读取不同的文件, 示例中会有说明。

2.2 辅助方法

很多示例都会用到共用的方法readFromInputStream, 该方法将InputStream转化String

  1. private String readFromInputStream(InputStream inputStream)
  2. throws IOException {
  3. StringBuilder resultStringBuilder = new StringBuilder();
  4. try (BufferedReader br
  5. = new BufferedReader(new InputStreamReader(inputStream))) {
  6. String line;
  7. while ((line = br.readLine()) != null) {
  8. resultStringBuilder.append(line).append("\n");
  9. }
  10. }
  11. return resultStringBuilder.toString();
  12. }

3.从Classpath读取文件

3.1 使用标准Java

src/main/resources读取文件fileTest.txt

  1. @Test
  2. public void test() throws IOException {
  3. String expectedData = "Hello,World!";
  4. Class<ReadFileTest> clazz = ReadFileTest.class;
  5. InputStream inputStream = clazz.getResourceAsStream("/fileTest.txt");
  6. String data = readFromInputStream(inputStream);
  7. Assert.assertThat(data, containsString(expectedData));
  8. }

在上面的代码中,我们通过当前类的getResourceAsStream方法加载文件,入参是绝对路径。

ClassLoader中相同的方法也可以使用。

  1. ClassLoader classLoader = getClass().getClassLoader();
  2. InputStream inputStream = classLoader.getResourceAsStream("fileTest.txt");
  3. String data = readFromInputStream(inputStream);

这两种方法的主要区别是, 当前类的ClassLoadergetResourceAsStream方法,入参路径是从classpath开始。

而类实例的入参是相对于包路径,但路径开始使用'/'符号, 也是绝对路径。

特别要注意的是, 文件打开读取完数据后, 始终需要关闭

inputStream.close();

3.2 使用commons-io库

另一个比较常用的方法是使用commons-io包里的FileUtils.readFileToString方法。

  1. <dependency>
  2. <groupId>commons-io</groupId>
  3. <artifactId>commons-io</artifactId>
  4. <version>1.4</version>
  5. </dependency>
  1. @Test
  2. public void useCommonIO() throws IOException {
  3. String expectedData = "Hello,World!";
  4. ClassLoader classLoader = getClass().getClassLoader();
  5. File file = new File(classLoader.getResource("fileTest.txt").getFile());
  6. String data = FileUtils.readFileToString(file, "UTF-8");
  7. assertEquals(expectedData, data.trim());
  8. }

该方法入参是File对象。这个工具类的优点是不用编码InputStream实例的相关代码。
这个库还提供了IOUtils类。

  1. @Test
  2. public void useCommonIO2() throws IOException {
  3. String expectedData = "Hello,World!";
  4. FileInputStream fis = new FileInputStream("src/test/resources/fileTest.txt");
  5. String data = IOUtils.toString(fis, "UTF-8");
  6. assertEquals(expectedData, data.trim());
  7. }

4.BufferedReader

  1. @Test
  2. public void bufferedReader() throws IOException {
  3. String expected_value = "Hello,World!";
  4. String file ="src/test/resources/fileTest.txt";
  5. BufferedReader reader = new BufferedReader(new FileReader(file));
  6. String currentLine = reader.readLine();
  7. reader.close();
  8. assertEquals(expected_value, currentLine);
  9. }

当读取结束的时候, reader.readLine()会返回null

5.Java NIO

NIO是在JDK7中添加。

5.1读取小文件

首先看一下关于Files.readAllLines的示例

  1. @Test
  2. public void readSmallFile() throws IOException {
  3. String expected_value = "Hello,World!";
  4. Path path = Paths.get("src/test/resources/fileTest.txt");
  5. String read = Files.readAllLines(path).get(0);
  6. assertEquals(expected_value, read);
  7. }

入参Path对象,Path可以认为是java.io.File的升级版本,提供一些额外的功能。

如果读取的是二进制文件,可以使用Files.readAllBytes()方法

5.2读取大文件

如果想要读取大文件, 我们可以使用Files类和BufferedReader类。

  1. @Test
  2. public void readLargeFile() throws IOException {
  3. String expected_value = "Hello,World!";
  4. Path path = Paths.get("src/test/resources/fileTest.txt");
  5. BufferedReader reader = Files.newBufferedReader(path);
  6. String line = reader.readLine();
  7. assertEquals(expected_value, line);
  8. }

5.3Files.lines

在JDK8中,Files类增加了lines方法,这个方法将返回Stream<String>。跟文件操作一样,Stream需要显式调用的close()。Files API提供了很多简单读取文件的方法。

6.Scanner

下面我们将使用Scanner读取文件,使用逗号(,)作为定界符(delimiter)。

  1. @Test
  2. public void whenReadWithScanner_thenCorrect()
  3. throws IOException {
  4. String file = "src/test/resources/fileTest.txt";
  5. Scanner scanner = new Scanner(new File(file));
  6. scanner.useDelimiter(",");
  7. assertTrue(scanner.hasNext());
  8. assertEquals("Hello", scanner.next());
  9. assertEquals("World!", scanner.next());
  10. scanner.close();
  11. }

Scanner默认的定界符是空格。它适用于从控制台读取输入或者内容有固定定界符的内容时。

7.StreamTokenizer

tokenizer会指出下一个token的类型,String或Number。

tokenizer.nval - 如果类型为Number时,读取该字段

tokenizer.sval - 如果类型为String时,读取该字段

  1. @Test
  2. public void readWithTokenize()
  3. throws IOException {
  4. String file = "src/test/resources/fileTestTokenizer.txt";
  5. FileReader reader = new FileReader(file);
  6. StreamTokenizer tokenizer = new StreamTokenizer(reader);
  7. // 1
  8. tokenizer.nextToken();
  9. assertEquals(StreamTokenizer.TT_WORD, tokenizer.ttype);
  10. assertEquals("Hello", tokenizer.sval);
  11. // 2
  12. tokenizer.nextToken();
  13. assertEquals(StreamTokenizer.TT_NUMBER, tokenizer.ttype);
  14. assertEquals(1, tokenizer.nval, 0.0000001);
  15. // 3
  16. tokenizer.nextToken();
  17. assertEquals(StreamTokenizer.TT_EOF, tokenizer.ttype);
  18. reader.close();
  19. }

8.DataInputStream

如果要读取二进制文件或者原生数据,可以使用DataInputStream

  1. @Test
  2. public void whenReadWithDataInputStream() throws IOException {
  3. String expectedValue = "Hello,World!";
  4. String file ="src/test/resources/fileTest.txt";
  5. String result = null;
  6. DataInputStream reader = new DataInputStream(new FileInputStream(file));
  7. int nBytesToRead = reader.available();
  8. if(nBytesToRead > 0) {
  9. byte[] bytes = new byte[nBytesToRead];
  10. reader.read(bytes);
  11. result = new String(bytes);
  12. }
  13. assertEquals(expectedValue, result);
  14. }

9.FileChannel

如果读取的是一个大文件,FileChannel速度会超过standard IO。

  1. @Test
  2. public void whenReadWithFileChannel()
  3. throws IOException {
  4. String expected_value = "Hello,World!";
  5. String file = "src/test/resources/fileTest.txt";
  6. RandomAccessFile reader = new RandomAccessFile(file, "r");
  7. FileChannel channel = reader.getChannel();
  8. int bufferSize = 1024;
  9. if (bufferSize > channel.size()) {
  10. bufferSize = (int) channel.size();
  11. }
  12. ByteBuffer buff = ByteBuffer.allocate(bufferSize);
  13. channel.read(buff);
  14. buff.flip();
  15. assertEquals(expected_value, new String(buff.array()));
  16. channel.close();
  17. reader.close();
  18. }

10.读取utf-8编码的文件

  1. @Test
  2. public void whenReadUTFEncodedFile()
  3. throws IOException {
  4. String expected_value = "你好,世界!";
  5. String file = "src/test/resources/fileTestUtf8.txt";
  6. BufferedReader reader = new BufferedReader
  7. (new InputStreamReader(new FileInputStream(file), "UTF-8"));
  8. String currentLine = reader.readLine();
  9. reader.close();
  10. assertEquals(expected_value, currentLine);
  11. }

11.从URL读取数据

  1. @Test
  2. public void readFromURL() throws IOException {
  3. URL urlObject = new URL("https://www.baidu.com");
  4. URLConnection urlConnection = urlObject.openConnection();
  5. InputStream inputStream = urlConnection.getInputStream();
  6. String data = readFromInputStream(inputStream);
  7. }

12.从jar包中读取文件

我们的目标是读取junit-4.12.jar包中的LICENSE-junit.txt文件。clazz只需要这个Jar中的类就行。

  1. @Test
  2. public void readFromJar() throws IOException {
  3. String expectedData = "Eclipse Public License";
  4. Class clazz = Test.class;
  5. InputStream inputStream = clazz.getResourceAsStream("/LICENSE-junit.txt");
  6. String data = readFromInputStream(inputStream);
  7. Assert.assertThat(data, containsString(expectedData));
  8. }

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/AllinToyou/article/detail/663913
推荐阅读
相关标签
  

闽ICP备14008679号