20.01_IO流(IO流概述及其分类)
-
1.概念
- IO流用来处理设备之间的数据传输
- Java对数据的操作是通过流的方式
- Java用于操作流的类都在IO包中
- 流按流向分为两种:输入流,输出流。
- 流按操作类型分为两种:
- 字节流 : 字节流可以操作任何数据,因为在计算机中任何数据都是以字节的形式存储的
- 字符流 : 字符流只能操作纯字符数据,比较方便。
-
2.IO流常用父类
- 字节流的抽象父类:
- InputStream
- OutputStream
- 字符流的抽象父类:
- Reader
- Writer
- 字节流的抽象父类:
-
3.IO程序书写
- 使用前,导入IO包中的类
- 使用时,进行IO异常处理
- 使用后,释放资源
20.02_IO流(FileInputStream)
read()一次读取一个字节
-
FileInputStream fis = new FileInputStream("aaa.txt"); //创建一个文件输入流对象,并关联aaa.txtint b; //定义变量,记录每次读到的字节while((b = fis.read()) != -1) { //从硬盘上读取一个个字节,将每次读到的字节赋值给b并判断是否是-1,-1是文件的结束标记System.out.println(b); //打印每一个字节}fis.close(); //关闭流,释放资源
20.03_IO流(read()方法返回值为什么是int)
read()方法读取的是一个字节,为什么返回是int,而不是byte
-
因为 字节输入流 可以操作任意类型的文件,比如图片音频等,这些文件底层都是以二进制形式的存储的,如果每次读取都返回byte,有可能在读到中间的时候遇到111111111那么这11111111是byte类型的-1,我们的程序是遇到-1就会停止不读了,后面的数据就读不到了,所以在读取的时候用int类型接收,如果11111111会在其前面补上24个0凑足4个字节(写入时的write方法会将添加的0弹出,保证写入时不出错,保证数据原样性),那么byte类型的-1就变成int类型的255了这样可以保证整个数据读完,而结束标记的-1就是int类型* 00010100 00100100 01000001 11111111 0000000* * 10000001 byte类型-1的原码* 11111110 -1的反码* 11111111 -1的补码* * 00000000 00000000 00000000 11111111
20.04_IO流(FileOutputStream)
- write()一次写出一个字节
20.05_IO流(FileOutputStream追加)
-
A:案例演示
- FileOutputStream的构造方法写出数据如何实现数据的追加写入
- FileOutputStream在创建对象的时候是:如果没有这个文件会帮我创建出来,如果有这个文件就会先将文件清空
-
public static void main(String[] args) throws IOException {//demo1();FileOutputStream fos = new FileOutputStream("yyy.txt",true); //如果想续写,就在第二个参数传truefos.write(97);fos.write(98);fos.close(); //abcab}public static void demo1() throws FileNotFoundException, IOException {FileOutputStream fos = new FileOutputStream("bbb.txt"); //创建字节输出流对象,如果没有bbb.txt,会创建出一个//fos.write(97); //a,虽然写出的是一个int数,但是在写出的时候会将前面的24个0去掉,所以写出的是一个bytefos.write(98); //bfos.write(99); //cfos.close();}
20.06_IO流(拷贝图片)
FileInputStream读取
FileOutputStream写出
/*** @param args* @throws IOException */public static void main(String[] args) throws IOException {//demo1();//demo2();//demo3();}public static void demo3() throws FileNotFoundException, IOException {//第二种拷贝,不推荐使用,因为有可能会导致内存溢出FileInputStream fis = new FileInputStream("致青春.mp3"); //创建输入流对象,关联致青春.mp3FileOutputStream fos = new FileOutputStream("copy.mp3"); //创建输出流对象,关联copy.mp3//int len = fis.available();//System.out.println(len); //4696468byte[] arr = new byte[fis.available()]; //创建与文件一样大小的字节数组fis.read(arr); //将文件上的字节读取到内存中fos.write(arr); //将字节数组中的字节数据写到文件上fis.close();fos.close();}public static void demo2() throws FileNotFoundException, IOException {FileInputStream fis = new FileInputStream("致青春.mp3"); FileOutputStream fos = new FileOutputStream("copy.mp3"); int b;while((b = fis.read()) != -1) { //在不断的读取每一个字节fos.write(b); //将每一个字节写出}fis.close(); //关流释放资源fos.close(); //按字节来,效率太低}public static void demo1() throws FileNotFoundException, IOException {FileInputStream fis = new FileInputStream("双元.jpg"); //创建输入流对象,关联双元.jpgFileOutputStream fos = new FileOutputStream("copy.jpg"); //创建输出流对象,关联copy.jpgint b;while((b = fis.read()) != -1) { //在不断的读取每一个字节fos.write(b); //将每一个字节写出}fis.close(); //关流释放资源fos.close();}
20.07_IO流(拷贝音频文件画原理图)
-
A:案例演示
- 字节流一次读写一个字节复制音频
- 弊端:效率太低
20.08_IO流(字节数组拷贝之available()方法)
-
A:案例演示
- int read(byte[] b):一次读取一个字节数组
- write(byte[] b):一次写出一个字节数组
- available()获取读的文件所有的字节个数
- 弊端:有可能会内存溢出
20.09_IO流(定义小数组)
write(byte[] b)
write(byte[] b, int off, int len)写出有效的字节个数
/*** @param args* 第三种拷贝* 定义小数组* @throws IOException */public static void main(String[] args) throws IOException {//demo1();//demo2();FileInputStream fis = new FileInputStream("致青春.mp3");FileOutputStream fos = new FileOutputStream("copy.mp3");byte[] arr = new byte[1024 * 8]; //8k,1024bytes=1k,小数组一般是1024的整数倍int len;while((len = fis.read(arr)) != -1) { //如果忘记加arr,返回的就不是要读取的字节个数(3个),而是字节的码表值(97,98,99),文件内会写入294个字节fos.write(arr,0,len);}fis.close();fos.close();}public static void demo2() throws FileNotFoundException, IOException {FileInputStream fis = new FileInputStream("xxx.txt");FileOutputStream fos = new FileOutputStream("yyy.txt");byte[] arr = new byte[2];int len;while((len = fis.read(arr)) != -1) { //以demo1为例,len第一次为2,第二次为1,若有第三次,没有数据,则返回-1//fos.write(arr); //abcbfos.write(arr,0,len); //abc}fis.close();fos.close();}public static void demo1() throws FileNotFoundException, IOException {FileInputStream fis = new FileInputStream("xxx.txt");byte[] arr = new byte[2]; //存储2个元素int a = fis.read(arr); //将文件上的字节读取到字节数组中System.out.println(a); //2,读到的有效字节个数for (byte b : arr) { //第一次获取到文件上的a(97)和b(98)System.out.println(b);}System.out.println("-----------------------");int c = fis.read(arr);System.out.println(c); //1(一共3个:abc),读到的有效字节个数for (byte b : arr) {System.out.println(b); //第二次获取到文件上的c(99)和b(98),数组中a的位置被c覆盖,而b还保留着}fis.close();}
20.10_IO流(定义小数组的标准格式)
-
A:案例演示
- 字节流一次读写一个字节数组复制图片和视频
20.11_IO流(BufferedInputStream和BufferOutputStream拷贝)
-
A:缓冲思想
- 字节流一次读写一个数组的速度明显比一次读写一个字节的速度快很多,
- 这是加入了数组这样的缓冲区效果,java本身在设计的时候,
- 也考虑到了这样的设计思想(装饰设计模式后面讲解),所以提供了字节缓冲区流
-
B.BufferedInputStream
- BufferedInputStream内置了一个缓冲区(数组)
- 从BufferedInputStream中读取一个字节时
- BufferedInputStream会一次性从文件中读取8192个, 存在缓冲区中, 返回给程序一个
- 程序再次读取时, 就不用找文件了, 直接从缓冲区中获取
- 直到缓冲区中所有的都被使用过, 才重新从文件中读取8192个
-
C.BufferedOutputStream
- BufferedOutputStream也内置了一个缓冲区(数组)
- 程序向流中写出字节时, 不会直接写到文件, 先写到缓冲区中
- 直到缓冲区写满, BufferedOutputStream才会把缓冲区中的数据一次性写到文件里
-
D.拷贝的代码(见下)
-
E.小数组的读写和带Buffered的读取哪个更快?
- 定义小数组如果是8192个字节大小和Buffered比较的话(两者都在内存中)
- 定义小数组会略胜一筹,因为读和写操作的是同一个数组
- 而Buffered操作的是两个数组
20.12_IO流(flush和close方法的区别)
-
flush()方法
- 用来刷新缓冲区的,刷新后可以再次写出
-
close()方法
-
用来关闭流释放资源的的,如果是带缓冲区的流对象的close()方法,不但会关闭流,还会再关闭流之前刷新缓冲区,关闭后不能再写出
/*** @param args* @throws IOException * close方法* 具备刷新的功能,在关闭流之前,就会先刷新一次缓冲区,将缓冲区的字节全都刷新到文件上(因为不可能每次都是8192个字节,会有余留),再关闭,close方法刷完之后就不能写了* flush方法* 具备刷新的功能,刷完之后还可以继续写*/public static void main(String[] args) throws IOException {//demo1();//flush和close方法的区别BufferedInputStream bis = new BufferedInputStream(new FileInputStream("致青春.mp3"));BufferedOutputStream bos = new BufferedOutputStream(new FileOutputStream("copy.mp3"));int b;while((b = bis.read()) != -1) {bos.write(b); //若忘记关流,copy的文件会变小}bis.close();bos.close();//bos.flush(); //copy的文件大小正常}public static void demo1() throws FileNotFoundException, IOException {FileInputStream fis = new FileInputStream("致青春.mp3"); //创建输入流对象,关联致青春.mp3FileOutputStream fos = new FileOutputStream("copy.mp3"); //创建输出流对象,关联copy.mp3//Buffered...在内存中进行,运算效率比硬盘高得多;Buffered...的内部方法:byte[] arr =new byte[8192];一次写8192个字节BufferedInputStream bis = new BufferedInputStream(fis); //创建缓冲区对象(对fis装饰),对输入流进行包装让其变得更加强大,fis也可以直接用匿名对象代替BufferedOutputStream bos = new BufferedOutputStream(fos); //创建缓冲区对象(对fos装饰)int b;while((b = bis.read()) != -1) {bos.write(b);}bis.close(); //缓冲区对象是输入流包装后的结果,故只需关闭缓冲对象,输入流也会被关闭bos.close();}
-
20.13_IO流(字节流读写中文)
-
字节流读取中文的问题
- 字节流在读中文的时候有可能会读到半个中文,造成乱码
-
字节流写出中文的问题
-
字节流直接操作的字节,所以写出中文必须将**字符串转换成字节数组 **
-
写出回车换行 write("\r\n".getBytes());
public static void main(String[] args) throws IOException {//demo1();FileOutputStream fos = new FileOutputStream("zzz.txt");fos.write("我读书少,你不要骗我".getBytes()); //转为字节数组,读没有问题fos.write("\r\n".getBytes()); //回车换行,在eclipse中会显示第二行行号,去掉就没有fos.close();}public static void demo1() throws FileNotFoundException, IOException {FileInputStream fis = new FileInputStream("yyy.txt");byte[] arr = new byte[4]; //一个中文代表两个字节,但可能有逗号,难确定int len;while((len = fis.read(arr)) != -1) {System.out.println(new String(arr,0,len)); //将字节数组转化为字符串}fis.close();}
-
20.14_IO流(流的 标准处理异常代码 1.6版本及其以前)
try finally嵌套
FileInputStream fis = null; //try中的fis,fos只在try中有效,若在此不赋值为null,fis.close 和fos.close会报错FileOutputStream fos = null;try {fis = new FileInputStream("aaa.txt");fos = new FileOutputStream("bbb.txt");int b;while((b = fis.read()) != -1) {fos.write(b);}} finally { //有try才能用finallytry {if(fis != null)fis.close();}finally {if(fos != null)fos.close(); //能关一个是一个,若上一个出问题了,就将这个关掉}}
20.15_IO流(流的标准处理异常代码1.7版本)
try close
try( //此处是小括号,将流括起来FileInputStream fis = new FileInputStream("aaa.txt");FileOutputStream fos = new FileOutputStream("bbb.txt");){ //大括号内放读写代码int b;while((b = fis.read()) != -1) {fos.write(b); //大括号执行完后,有自动关闭流功能,不需要再去写关闭流的语句}}class MyClose implements AutoCloseable {public void close() {System.out.println("我关了"); //实现了AutoCloseable接口,重写了close()方法,即使上面没有调用语句,也会执行,不过没有必要,只是演示}
- 原理
- 在try()中创建的流对象必须实现了AutoCloseable这个接口,如果实现了,在try后面的{}(读写代码)执行后就会自动调用,流对象的close方法将流关掉
20.16_IO流(图片加密)
给图片加密
/*** @param args* @throws IOException * 将写出的字节异或上一个数,这个数就是密钥,解密的时候再次异或就可以了*/public static void main(String[] args) throws IOException {BufferedInputStream bis = new BufferedInputStream(new FileInputStream("copy.jpg"));BufferedOutputStream bos = new BufferedOutputStream(new FileOutputStream("copy2.jpg"));int b;while((b = bis.read()) != -1) {bos.write(b ^ 123); //异或两次等于本身}bis.close();bos.close();}
20.17_IO流(拷贝文件)
在控制台录入文件的路径,将文件拷贝到当前项目下(如eclipse的Day 20文件夹下)
/*** 分析:* * 1,定义方法对键盘录入的路径进行判断,如果是文件就返回* 2,在主方法中接收该文件* 3,读和写该文件* @throws IOException */public static void main(String[] args) throws IOException {File file = getFile(); //创建键盘录入对象,获取文件BufferedInputStream bis = new BufferedInputStream(new FileInputStream(file));BufferedOutputStream bos = new BufferedOutputStream(new FileOutputStream(file.getName())); //名称不变int b;while((b = bis.read()) != -1) {bos.write(b);}bis.close();bos.close();}/** 定义一个方法获取键盘录入的文件路径,并封装成File对象返回* 1,返回值类型File* 2,参数列表无*/public static File getFile() {Scanner sc = new Scanner(System.in); //创建键盘录入对象System.out.println("请输入一个文件的路径:");while(true) { //若不加while语句,会报返回类型错误String line = sc.nextLine(); //接收键盘录入的路径File file = new File(line); //封装成File对象,并对其进行判断if(!file.exists()) { //文件不存在System.out.println("您录入的文件路径不存在,请重新录入:");}else if(file.isDirectory()) { //是文件夹而不是文件System.out.println("您录入的是文件夹路径,请重新录入:");}else {return file;}}}
20.18_IO流(录入数据拷贝到文件)
将键盘录入的数据拷贝到当前项目下的text.txt文件中,键盘录入数据当遇到quit时就退出
/*** @throws IOException */public static void main(String[] args) throws IOException {//1,创建键盘录入对象Scanner sc = new Scanner(System.in);//2,创建输出流对象,关联text.txt文件FileOutputStream fos = new FileOutputStream("text.txt");System.out.println("请输入数据:");//3,定义无限循环while(true) {String line = sc.nextLine(); //将键盘录入的数据存储在line中//4,遇到quit退出循环if("quit".equals(line)) {break; //quit要换行之后输入才有用}//5,如果不quit,就将内容写出fos.write(line.getBytes()); //字符串写出必须转换成字节数组fos.write("\r\n".getBytes());}//6,关闭流fos.close();}