Java IO详解(五)------包装流

 File 类的介绍:https://cloud.tencent.com/developer/article/1012532

Java IO 流的分类介绍:https://cloud.tencent.com/developer/article/1012539

Java IO 字节输入输出流:https://cloud.tencent.com/developer/article/1012565

Java IO 字符输入输出流:https://cloud.tencent.com/developer/article/1012570

  我们在 Java IO 流的分类介绍  这篇博客中介绍知道:

  根据功能分为节点流和包装流(处理流)

    节点流:可以从或向一个特定的地方(节点)读写数据。如FileReader.

    处理流:是对一个已存在的流的连接和封装,通过所封装的流的功能调用实现数据读写。如BufferedReader.处理流的构造方法总是要带一个其他的流对象做参数。一个流对象经过其他流的多次包装,称为流的链接。

1、前面讲的字符输入输出流,字节输入输出流都是字节流。那么什么是包装流呢?

  ①、包装流隐藏了底层节点流的差异,并对外提供了更方便的输入\输出功能,让我们只关心这个高级流的操作

  ②、使用包装流包装了节点流,程序直接操作包装流,而底层还是节点流和IO设备操作

  ③、关闭包装流的时候,只需要关闭包装流即可

2、缓冲流

  缓冲流:是一个包装流,目的是缓存作用,加快读取和写入数据的速度。

  字节缓冲流:BufferedInputStream、BufferedOutputStream

  字符缓冲流:BufferedReader、BufferedWriter

案情回放:我们在将字符输入输出流、字节输入输出流的时候,读取操作,通常都会定义一个字节或字符数组,将读取/写入的数据先存放到这个数组里面,然后在取数组里面的数据。这比我们一个一个的读取/写入数据要快很多,而这也就是缓冲流的由来。只不过缓冲流里面定义了一个 数组用来存储我们读取/写入的数据,当内部定义的数组满了(注意:我们操作的时候外部还是会定义一个小的数组,小数组放入到内部数组中),就会进行下一步操作。 

下面是没有用缓冲流的操作:   

//1、创建目标对象,输入流表示那个文件的数据保存到程序中。不写盘符,默认该文件是在该项目的根目录下
			//a.txt 保存的文件内容为:AAaBCDEF
		File target = new File("io"+File.separator+"a.txt");
		//2、创建输入流对象
		InputStream in = new FileInputStream(target);
		//3、具体的 IO 操作(读取 a.txt 文件中的数据到程序中)
			/**
			 * 注意:读取文件中的数据,读到最后没有数据时,返回-1
			 * 	int read():读取一个字节,返回读取的字节
			 * 	int read(byte[] b):读取多个字节,并保存到数组 b 中,从数组 b 的索引为 0 的位置开始存储,返回读取了几个字节
			 * 	int read(byte[] b,int off,int len):读取多个字节,并存储到数组 b 中,从数组b 的索引为 0 的位置开始,长度为len个字节
			 */
		//int read():读取一个字节,返回读取的字节
		int data1 = in.read();//获取 a.txt 文件中的数据的第一个字节
		System.out.println((char)data1); //A
		//int read(byte[] b):读取多个字节保存到数组b 中
		byte[] buffer  = new byte[10];//这里我们定义了一个 长度为 10 的字节数组,用来存储读取的数据
		in.read(buffer);//获取 a.txt 文件中的前10 个字节,并存储到 buffer 数组中
		System.out.println(Arrays.toString(buffer)); //[65, 97, 66, 67, 68, 69, 70, 0, 0, 0]
		System.out.println(new String(buffer)); //AaBCDEF[][][]
		
		//int read(byte[] b,int off,int len):读取多个字节,并存储到数组 b 中,从索引 off 开始到 len
		in.read(buffer, 0, 3);
		System.out.println(Arrays.toString(buffer)); //[65, 97, 66, 0, 0, 0, 0, 0, 0, 0]
		System.out.println(new String(buffer)); //AaB[][][][][][][]
		//4、关闭流资源
		in.close();

  我们查看 缓冲流的 JDK 底层源码,可以看到,程序中定义了这样的 缓存数组,大小为 8192

BufferedInputStream:

BufferedOutputStream:

//字节缓冲输入流
		BufferedInputStream bis = new BufferedInputStream(
				new FileInputStream("io"+File.separator+"a.txt"));
		//定义一个字节数组,用来存储数据
		byte[] buffer = new byte[1024];
		int len = -1;//定义一个整数,表示读取的字节数
		while((len=bis.read(buffer))!=-1){
			System.out.println(new String(buffer,0,len));
		}
		//关闭流资源
		bis.close();
		
		//字节缓冲输出流
		BufferedOutputStream bos = new BufferedOutputStream(
				new FileOutputStream("io"+File.separator+"a.txt"));
		bos.write("ABCD".getBytes());
		bos.close();
//字符缓冲输入流
		BufferedReader br = new BufferedReader(
				new FileReader("io"+File.separator+"a.txt"));
		char[] buffer = new char[10];
		int len = -1;
		while((len=br.read(buffer))!=-1){
			System.out.println(new String(buffer,0,len));
		}
		br.close();
		
		//字符缓冲输出流
		BufferedWriter bw = new BufferedWriter(
				new FileWriter("io"+File.separator+"a.txt"));
		bw.write("ABCD");
		bw.close();

 3、转换流:把字节流转换为字符流

  InputStreamReader:把字节输入流转换为字符输入流

  OutputStreamWriter:把字节输出流转换为字符输出流

 用转换流进行文件的复制:

/**
		 * 将 a.txt 文件 复制到 b.txt 中
		 */
		//1、创建源和目标
		File srcFile = new File("io"+File.separator+"a.txt");
		File descFile = new File("io"+File.separator+"b.txt");
		//2、创建字节输入输出流对象
		InputStream in = new FileInputStream(srcFile);
		OutputStream out = new FileOutputStream(descFile);
		//3、创建转换输入输出对象
		Reader rd = new InputStreamReader(in);
		Writer wt = new OutputStreamWriter(out);
		//3、读取和写入操作
		char[] buffer = new char[10];//创建一个容量为 10 的字符数组,存储已经读取的数据
		int len = -1;//表示已经读取了多少个字符,如果是 -1,表示已经读取到文件的末尾
		while((len=rd.read(buffer))!=-1){
			wt.write(buffer, 0, len);
		}
		//4、关闭流资源
		rd.close();
		wt.close();

 4、内存流(数组流):

把数据先临时存在数组中,也就是内存中。所以关闭 内存流是无效的,关闭后还是可以调用这个类的方法。底层源码的 close()是一个空方法

  ①、字节内存流:ByteArrayOutputStream 、ByteArrayInputStream

//字节数组输出流:程序---》内存
		ByteArrayOutputStream bos = new ByteArrayOutputStream();
		//将数据写入到内存中
		bos.write("ABCD".getBytes());
		//创建一个新分配的字节数组。 其大小是此输出流的当前大小,缓冲区的有效内容已被复制到其中。 
		byte[] temp = bos.toByteArray();
		System.out.println(new String(temp,0,temp.length));
		
		byte[] buffer = new byte[10];
		///字节数组输入流:内存---》程序
		ByteArrayInputStream bis = new ByteArrayInputStream(temp);
		int len = -1;
		while((len=bis.read(buffer))!=-1){
			System.out.println(new String(buffer,0,len));
		}
		
		//这里不写也没事,因为源码中的 close()是一个空的方法体
		bos.close();
		bis.close();

  ②、字符内存流:CharArrayReader、CharArrayWriter

//字符数组输出流
		CharArrayWriter caw = new CharArrayWriter();
		caw.write("ABCD");
		//返回内存数据的副本
		char[] temp = caw.toCharArray();
		System.out.println(new String(temp));
		
		//字符数组输入流
		CharArrayReader car = new CharArrayReader(temp);
		char[] buffer = new char[10];
		int len = -1;
		while((len=car.read(buffer))!=-1){
			System.out.println(new String(buffer,0,len));
		}

  ③、字符串流:StringReader,StringWriter(把数据临时存储到字符串中)

//字符串输出流,底层采用 StringBuffer 进行拼接
		StringWriter sw = new StringWriter();
		sw.write("ABCD");
		sw.write("帅锅");
		System.out.println(sw.toString());//ABCD帅锅

		//字符串输入流
		StringReader sr = new StringReader(sw.toString());
		char[] buffer = new char[10];
		int len = -1;
		while((len=sr.read(buffer))!=-1){
			System.out.println(new String(buffer,0,len));//ABCD帅锅
		}

5、合并流:把多个输入流合并为一个流,也叫顺序流,因为在读取的时候是先读第一个,读完了在读下面一个流。

//定义字节输入合并流
		SequenceInputStream seinput = new SequenceInputStream(
				new FileInputStream("io/a.txt"), new FileInputStream("io/b.txt"));
		byte[] buffer = new byte[10];
		int len = -1;
		while((len=seinput.read(buffer))!=-1){
			System.out.println(new String(buffer,0,len));
		}
		
		seinput.close();

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏Java爬坑系列

【Java入门提高篇】Day17 Java异常处理(下)

  今天继续讲解java中的异常处理机制,主要介绍Exception家族的主要成员,自定义异常,以及异常处理的正确姿势。

34515
来自专栏Java爬坑系列

【Java入门提高篇】Day17 Java异常处理(下)

1083
来自专栏Java帮帮-微信公众号-技术文章全总结

【大牛经验】探讨Java的异常与错误处理

探讨Java的异常与错误处理 ENTER TITLE ? Java中的异常处理机制已经比较成熟,我们的Java程序到处充满了异常的可能,如果对这些异常不做预先的...

3616
来自专栏JackieZheng

学习SpringMVC——拦截器

  拦截器,顾名思义就是用来拦截的。   那什么是拦截,又为什么要拦截。对于Spring MVC来说,拦截器主要的工作对象就是用户的请求,拦截下来之后,我们可以...

1978
来自专栏程序你好

C# 7.3新特性一览

853
来自专栏java一日一条

一个Java对象到底占用多大内存

大家可以用这个代码边看边验证,注意的是,运行这个程序需要通过javaagent注入Instrumentation,具体可以看原博客。我今天主要是总结下手动计算J...

421
来自专栏有趣的Python

8-Java常用工具类-输入输出流

生活中无处不在,只要涉及到传输。复制粘贴操作;修改头像,将本地数据上传到网络服务器。

681
来自专栏码匠的流水账

聊聊flink的ListCheckpointed

flink-streaming-java_2.11-1.7.0-sources.jar!/org/apache/flink/streaming/api/chec...

512
来自专栏青枫的专栏

装饰设计模式图解

装饰设计模式的概述     装饰模式就是使用被装饰类的一个子类的实例,在客户端将这个子类的实例交给装饰类。是继承的替代方案。 优点     使用...

361
来自专栏java学习

面试题7(考察运算符的优先级)

请选择下面代码运行后打印的结果。 public static void main(String[]args){ int x=5; int y=3; x=x+(x...

2928

扫码关注云+社区