A股上市公司传智教育(股票代码 003032)旗下技术交流社区北京昌平校区

 找回密码
 加入黑马

QQ登录

只需一步,快速开始

© 顾传文 中级黑马   /  2013-2-25 22:10  /  5691 人查看  /  3 人回复  /   0 人收藏 转载请遵从CC协议 禁止商业使用本文

本帖最后由 顾传文 于 2013-3-12 09:15 编辑

在读取流和输入流之后,都必须关闭流,避免占用资源。为什么ByteArrayInputStream和ByteArrayOutputStream可以不关闭呢?

评分

参与人数 1技术分 +1 收起 理由
Rancho_Gump + 1

查看全部评分

3 个回复

倒序浏览
因为ByteArrayInputStream和ByteArrayOutputStream 内部封装了缓冲区,不涉及底层资源调用,关闭流就是为了防止底层资源泄漏【内存】。他根本没有调用底层资源,所以关不关都一样

评分

参与人数 1技术分 +1 收起 理由
Rancho_Gump + 1

查看全部评分

回复 使用道具 举报
本帖最后由 王文正 于 2013-2-26 12:43 编辑

对于帖主问的这个问题 我觉得首先你要先明白ByteArrayInputStream和ByteArrayOutputStream含义和用法

ByteArrayOutputStream的用法如下:
我的个人理解是ByteArrayOutputStream是用来缓存数据的(数据写入的目标(output stream原义)),向它的内部缓冲区写入数据,缓冲区自动增长,当写入完成时可以从中提取数据。由于这个原因,ByteArrayOutputStream常用于存储数据以用于一次写入。
以下是JDK中的记载:  
public class ByteArrayOutputStream  extends OutputStream
    此类实现了一个输出流,其中的数据被写入一个 byte 数组。缓冲区会随着数据的不断写入而自动增长。可使用 toByteArray()和 toString()获取数据。
   关闭 ByteArrayOutputStream 无效。此类中的方法在关闭此流后仍可被调用,而不会产生任何IOException。
在这里我可以给你举出一个例子:
从文件中读取二进制数据,全部存储到ByteArrayOutputStream中。
FileInputStream fis=new FileInputStream("test");
BufferedInputStream bis=new BufferedInputStream(fis);
ByteArrayOutputStream baos=new ByteArrayOutputStream();
int c=bis.read();//读取bis流中的下一个字节
while(c!=-1){
     baos.write(c);
     c=bis.read();
}
bis.close();
byte retArr[]=baos.toByteArray();



ByteArrayInputStream的用法
相对而言,ByteArrayInputStream比较少见。先看JDK文档中的介绍:
public class ByteArrayInputStreamextends InputStreamByteArrayInputStream 包含一个内部缓冲区,该缓冲区包含从流中读取的字节。内部计数器跟踪 read 方法要提供的下一个字节。
关闭 ByteArrayInputStream 无效。此类中的方法在关闭此流后仍可被调用,而不会产生任何 IOException。
构造函数:
ByteArrayInputStream(byte[] buf)  
注意它需要提供一个byte数组作为缓冲区。
与大部分Inputstream的语义类似,可以从它的缓冲区中读取数据,所以我们可以在它的外面包装另一层的inputstream以使用我们需要的读取方法。
我个人认为一个比较好的用途是在网络中读取数据包,由于数据包一般是定长的,我们可以先分配一个够大的byte数组,比如byte buf[]=new byte[1024];
然后调用某个方法得到网络中的数据包,
例如:
Socket s=...;
DataInputStream dis=new DataInputStream(s.getInputStream());
dis.read(buf);//把所有数据存到buf中
ByteArrayInputStream bais=new ByteArrayInputStream(buf); //把刚才的部分视为输入流
DataInputStream dis_2=new DataInputStream(bais);
//现在可以使用dis_2的各种read方法,读取指定的字节
比如第一个字节是版本号,dis_2.readByte();
等等……
上面的示例的两次包装看上去有点多此一举,但使用ByteArrayInputStream的好处是关掉流之后它的数据仍然存在。


   

评分

参与人数 1技术分 +1 收起 理由
Rancho_Gump + 1

查看全部评分

回复 使用道具 举报
我从另外一个角度来说这个事儿吧,你close的目的是什么?释放流关联资源。字节数组输入输出流关联的资源是什么?它内部封装的字节数组。字节数组存在于哪儿?堆内存中。那么我close就是要释放堆内存中的数组对象了,但堆内存中的对象,在java中你能直接处理释放吗?不能,因为它是被垃圾回收机制自动管理的,我们没法直接对之操作。那么,即使你关闭了,这个字节数组在堆内存仍然存在的了?是的。所以你仍能可以使用啊。
这里的close其实是没有意义的,但因为它是InputStream和OutputStream的实现子类,所以要复写close方法,以遵守java的语法。
回复 使用道具 举报
您需要登录后才可以回帖 登录 | 加入黑马