java FileInputStream(输入流)
一、File流概念
JAVA中针对文件的读写操作设置了一系列的流,其中主要有FileInputStream,FileOutputStream,FileReader,FileWriter四种最为常用的流
二、FileInputStream
1. FileInputStream概念
FileInputStream流被称为文件字节输入流,意思指对文件数据以字节的形式进行读取操作如读取图片视频等
2. 构造方法
2.1 通过打开与File类对象代表的实际文件的链接来创建FileInputStream流对象
public FileInputStream(File file) throws FileNotFoundException{}
若File类对象的所代表的文件不存在;不是文件是目录;或者其他原因不能打开的话,则会抛出FileNotFoundException
/*** * 运行会产生异常并被扑捉--因为不存在xxxxxxxx这样的文件*/
public static void main(String[] args){File file=new File("xxxxxxxx"); //根据路径创建File类对象--这里路径即使错误也不会报错,因为只是产生File对象,还并未与计算机文件读写有关联try{FileInputStream fileInputStream=new FileInputStream(file);//与根据File类对象的所代表的实际文件建立链接创建fileInputStream对象}catch (FileNotFoundException e){System.out.println("文件不存在或者文件不可读或者文件是目录");} }
2.2 通过指定的字符串参数(路径)来创建File类对象,而后再与File对象所代表的实际路径建立链接创建FileInputStream流对象
public FileInputStream(String name) throws FileNotFoundException {this(name != null ? new File(name) : null);}
该方法实际调用的还是上面那个方法。
3. 相关方法
3.1 public int read() throws IOException
官方文档
public int read() throws IOException
从此输入流中读取一个数据字节。如果没有输入可用,则此方法将阻塞。
指定者:
类 InputStream 中的 read
返回:
下一个数据字节;如果已到达文件末尾,则返回 -1。
理解读取的字节为什么返回int型变量
从输入流中读取一个字节返回int型变量,若到达文件末尾,则返回-1
我们读取的字节实际是由8位二进制组成,二进制文件不利于直观查看,可以转成常用的十进制进行展示,因此需要把读取的字节从二进制转成十进制整数,故返回int型。
如果当文件未到底时,我们读取的是字节,若返回byte类型,那么势必造成同一方法返回类型不同的情况这是不允许的
既然说可以测试任意形式的文件,那么用两种不同格式的,测试文件data1.txt和data2.txt,里面均放入1个数字"1",两文件的格式分别为:ANSI和Unicode。
我们用如下代码测试
import java.io.FileInputStream;
import java.io.IOException;public class Test {public static void main(String[] args) throws IOException {FileInputStream fis = new FileInputStream("data1.txt");//ANSI格式for (int i = 0; i < 5; i++) {System.out.println(fis.read()); }fis.close(); System.out.println("------------------");fis = new FileInputStream("data2.txt");//Unicode格式for (int i = 0; i < 5; i++) {System.out.println(fis.read()); }fis.close();}
}
文件里不是只有一个数字吗,为什么循环5次,什么鬼?稍后知晓,先看输出结果:
49
-1
-1
-1
-1
------------------
255
254
49
0
-1
结果怎么会是这样呢?
1.因为ANSI编码没有文件头,因此数字字符1只占一个字节,并且1的Ascii码为49因此输出49,而Unicode格式有2个字节的文件头,并且以2个字节表示一个字符,对于Ascii字符对应的字符则是第2位补0,因此1的Unicode码的两位十进制分别为49和0;
附:文本文件各格式文件头:ANSI类型:什么都没有,UTF-8类型:EF BB BF,UNICODE类型:FF FE,UNICODE BIG ENDIAN类型:FE FF
2.从返回的结果来看,返回的是当前的字节数据,API文档中原文为:“下一个数据字节,如果已到达文件末尾,则返回 -1。”(英文原文为:the next byte of data, or -1 if the end of the file is reached),应该理解成:此时的指针在下一个数据字节的开始位置。
3.2 read (byte[ ] b )
官方文档
public int read(byte[] b) throws IOException
从此输入流中将最多 b.length 个字节的数据读入一个 byte 数组中。在某些输入可用之前,此方法将阻塞。
覆盖:
类 InputStream 中的 read
参数:
b - 存储读取数据的缓冲区。
返回:
读入缓冲区的字节总数,如果因为已经到达文件末尾而没有更多的数据,则返回 -1。
解读:
最多b.length个字节的数据读入一个byte数据组中,即,最多将byte数组b填满;
返回读入缓冲的字节总数,如果因为已经到达文件末尾而没有更多的数据,则返回-1。这里即这为朋友的问题点,为什么用-1来判断文件的结束。他的理由为,假设3个字节源数据,用2个字节的数组来缓存,当第2次读取的时候到达了文件的结尾,此时应该返回-1了,岂不是只读取到了2个字节?
同样,我们来测试:
测试文件,data.txt,文件格式ANSI,文件内容123,测试代码:
import java.io.FileInputStream;
import java.io.IOException;
import java.util.Arrays;public class Test {public static void main(String[] args) throws IOException {FileInputStream fis = new FileInputStream("data.txt");//ANSI格式byte[] b = new byte[2];for (int i = 0; i < 3; i++) {System.out.print("第"+(i+1)+"次读取返回的结果:"+fis.read(b));System.out.println(",读取后数组b的内容为:"+Arrays.toString(b));}fis.close();}
}
输出结果:
第1次读取返回的结果:2,读取后数组b的内容为:[49, 50]
第2次读取返回的结果:1,读取后数组b的内容为:[51, 50]
第3次读取返回的结果:-1,读取后数组b的内容为:[51, 50]
测试数据文件采用的是ANSI格式,放入3个数字,因此为3个字节,这里测试读3次,从代码中可以看出,b为一个byte数组,大小为2,即每次可以存放2个字节。那么问题来了,第一次读取的时候读到2个字节返回很好理解,而第2次的时候,由于只剩下一个字节,此处到了文件的结尾,按照朋友对API文档的理解,应该返回-1才对?
让我们看看源码吧
public int read(byte b[]) throws IOException {return readBytes(b, 0, b.length);}
private native int readBytes(byte b[], int off, int len) throws IOException;
晴天霹雳,是个被native修饰的方法,因此没办法继续一步看代码了。没啥好说的,用个代码类继承FileInputStream,覆盖read(byte b)方法,看代码即能理解:
import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.io.IOException;public class MyFileInputStream extends FileInputStream{public MyFileInputStream(String name) throws FileNotFoundException {super(name);}@Overridepublic int read(byte[] b) throws IOException {int getData = read();if (getData==-1) {return -1;}else{b[0] = (byte)getData;for (int i = 1; i < b.length; i++) {getData = read();if(-1==getData)return i;b[i] = (byte)getData;}}return b.length;}
}
原测试代码做小小的改动:
import java.io.FileInputStream;
import java.util.Arrays;public class Test {public static void main(String[] args) throws Exception {FileInputStream fis = new MyFileInputStream("data.txt");//ANSI格式byte[] b = new byte[2];for (int i = 0; i < 3; i++) {System.out.print("第"+(i+1)+"次读取返回的结果:"+fis.read(b));System.out.println(",读取后数组b的内容为:"+Arrays.toString(b));}fis.close();}
}
输出结果与原结果一致:
第1次读取返回的结果:2,读取后数组b的内容为:[49, 50]
第2次读取返回的结果:1,读取后数组b的内容为:[51, 50]
第3次读取返回的结果:-1,读取后数组b的内容为:[51, 50]
说明此方法内部调用的是read()方法一个个的来读数据。 如果第一个字节不是 -1 就会继续读,如果后续的read()返回的是-1 那么此方法就不会返回 - 1 返回的是i。
正确显示文本内容
import java.io.FileInputStream;public class Test {public static void main(String[] args) throws Exception {FileInputStream fis = new MyFileInputStream("data.txt");//ANSI格式byte[] b = new byte[2];int len ;while (-1!=(len = fis.read(b))) {System.out.println(new String(b,0,len));}fis.close();}
}
3.3 available()方法
public int available();
方法的返回类型为int,它返回在解除阻塞期间可以从此FileInputStream读取的剩余可用字节数。
java FileInputStream(输入流)相关推荐
- Java FileInputStream
1. Java FileInputStream类 (1. Java FileInputStream Class) Java FileInputStream class is a part of jav ...
- java输出流输入流的使用_Java中的IO流之文件输入输出流
Java中的IO流之文件输入输出流 1.文件流介绍 文件流是以字节为单位进行读写文件的,所以属于字节流,并且是低级流.文件流位于java.io包下. 输入输出流分别是FileInputSteam和Fi ...
- java中的输入流类,Java数据输入流类
Java DataInputStream类 数据输入流允许应用程序以与机器无关方式从底层输入流中读取基本 Java 数据类型. 下面的构造方法用来创建数据输入流对象. DataInputStream ...
- java fileinputstream.read(byte[])_Java分享笔记:FileInputStream流的 read()方法 和 read(byte[] b)方法...
1 /*------------------------ 2 FileInputStream: 3 ....//输入流,字节流 4 ....//从硬盘中存在的一个文件中读取内容,读取到程序中 5 .. ...
- java用输入流创建数据文件_java开发知识IO知识之输入输出流以及文件
java开发知识IO知识之输入输出流以及文件 一丶流概述 流十一组有序的数据序列.根据操作的类型,可以分为输入流跟输出流两种. IO(input/output)输入/输出流提供了一条通道程序.可以使用 ...
- java用输入流创建数据文件_Java实现文件写入——输入输出流(IO流)
输入输出含义和意义: 输入和输出功能是Java对程序处理数据能力的提高,java以流的形式处理数据. 流是一组有序的数据序列,根据操作类型,分为输入流和输出流. 程序从输入流读取数据,向输出流写入数据 ...
- java切割输入流_Java分割流,对象流,参与序列化,分割读取文件,对象输入或输出流等【诗书画唱】...
自己写的题目:用分割流把一个Java文件分割读取,打印出每次读取的内容,要求每次读取打印500个字节. package fenGeLiu; import java.io.*; public class ...
- java缓冲输入流_java入门 -- Java I/O(五)输入缓冲流BufferedInputStream
/* * 缓冲输入字节流: * 作用:提高输入效率,比缓冲字节流跟高 * 输入字节流体系: * ----| InputStream 抽象列 * ---------| FileInputStream 读 ...
- java 字节输入流_JavaIO流(一)-字节输入流与字符输入流
IO流详解 一.输入流 字节输入流 FileInputSteam 1.构造方法:public FileInputStream(File file) {} public FileInputStream( ...
最新文章
- 拥抱AI技术,赋能智慧工业
- python中plot的plt.text_用Python进行数据可视化的第一步,全面详解matplotlib中样式属性...
- getchar()和EOF
- 职场人必备的几个PPT进阶小技巧
- 数列递推(牛客练习赛83)(数学、分块)
- MySQL 读写分离 使用驱动com.mysql.jdbc.ReplicationDriver
- linux 离线安装中文,linux离线安装及配置redis-Go语言中文社区
- 带父节点的平衡二叉树_平衡二叉树 通俗易懂
- Halcon 学习总结——错误处理方法
- 项目-1.CROSSFORMER论文与代码解析(CrossFormer: A Versatile Vision Transformer Based on Cross-scale Attention)
- QT:QSS自定义QLineEdit实例
- html文本框柔化边缘,html – CSS模糊并使用绝对div保留锐利边缘
- 7张图让你看懂互联网营销思维与传统思维本质区别
- XX.sh和./XX.sh
- 软件测试缺陷表怎样描述,作为软件测试人员,如何描述缺陷(Defect)?
- 【java期末复习题】第4章 面向对象基础
- Java程序设计课程——实验2
- 设备管理系统html,设备管理系统有哪些功能?- 智造家
- 云效率的 3 个方面:成本、性能和速度
- java 将A对象的值copy到B对象(根据属性名设置属性值)