管道流

  • 管道流的主要作用是可以进行两个线程间的通讯,分为管道输出流(PipedOutputStream)、管道输入流(PipedInputStream),如果想要进行管道输出,则必须要把输出流连在输入流之上,在PipedOutputStream类上有如下的一个方法用于连接管道:

    public void connect(PipedInputStream snk)throws IOException
  • 通常是创建两个单独的线程来实现通信,如果是单个线程的话容易出现线程堵塞,因为输出流最多只能向缓冲区写入1024个字节的数据,如果超出就会出现线程堵塞,因此必须创建多个线程实现缓冲区的释放和存储

PipedOutputStream

  • 管道输出流是管道的发送端,可以将管道输出流连接到管道输入流来创建一个通信管道,通常,数据由某个线程写入 PipedOutputStream 对象,并由其他线程从连接的 PipedInputStream 读取。不建议对这两个对象尝试使用单个线程,因为这样可能会造成该线程死锁。如果某个线程正从连接的管道输入流中读取数据字节,但该线程不再处于活动状态,则该管道被视为处于 毁坏 状态。

构造函数

  • PipedOutputStream() 创建尚未连接到管道输入流的管道输出流。
  • PipedOutputStream(PipedInputStream snk) 创建连接到指定管道输入流的管道输出流。

常用函数

  • close() 关闭
  • void connect(PipedInputStream snk) 将此管道输出流连接到接收者。
  • void flush() 刷新此输出流并强制写出所有缓冲的输出字节。
  • void write(byte[] b, int off, int len)len 字节从初始偏移量为 off 的指定 byte 数组写入该管道输出流。
  • void write(int b) 将指定 byte 写入传送的输出流。

PipedInputStream

  • 管道输入流应该连接到管道输出流;管道输入流提供要写入管道输出流的所有数据字节。通常,数据由某个线程从 PipedInputStream 对象读取,并由其他线程将其写入到相应的 PipedOutputStream。不建议对这两个对象尝试使用单个线程,因为这样可能死锁线程。管道输入流包含一个缓冲区,可在缓冲区限定的范围内将读操作和写操作分离开。 如果向连接管道输出流提供数据字节的线程不再存在,则认为该管道已损坏。

构造函数

  • PipedInputStream() 创建尚未连接的 PipedInputStream
  • PipedInputStream(PipedOutputStream src) 创建 PipedInputStream,使其连接到管道输出流 src

常用函数

  • int available() 返回可以不受阻塞地从此输入流中读取的字节数。
  • void close() 关闭此管道输入流并释放与该流相关的所有系统资源。
  • void connect(PipedOutputStream src) 使此管道输入流连接到管道输出流 src
  • int read() 读取此管道输入流中的下一个数据字节。
  • int read(byte[] b, int off, int len) 将最多 len 个数据字节从此管道输入流读入 byte 数组。
  • protected void receive(int b) 接收数据字节。

实例

package IO;

import java.io.IOException;
import java.io.PipedInputStream;
import java.io.PipedOutputStream; /**
* Created by chenjiabing on 17-5-25.
*/ /**
* 注意的问题:
* 1.写线程正在往缓冲区写数据的时候,但是此时的读线程的管道结束,那么此时的写线程的管道就会发生IOException异常
* 2.读线程正在从缓冲区读数据的时候,但是此时的写线程的管道已经结束了,此时就会引起读线程的管道发生IOException异常
* 3.必须是启用多线程才能实现管道之间的读写,否则会出现堵塞现象,因为这里的PipeOutputStream每次向缓冲区写入的字节数最大是1024,如果不及时的减少缓冲区的数据量就会出现堵塞
*/ public class demo7 {
public static PipedOutputStream outputStream = new PipedOutputStream();
public static PipedInputStream inputStream = new PipedInputStream(); /**
* 创建一个写入数据进程,使用的是PipeOutStream,将数据写入管道中
*/
public static void send() {
new Thread(new Runnable() {
@Override
public void run() {
byte[] bytes = new byte[2000]; //创建一个2000字节的数组
while (true) {
try {
outputStream.write(bytes, 0, 2000); //写入管道,但是这里的缓冲区最多写入1024个字节的数据,因此这个是一次没有写完
System.out.println("写入成功");
} catch (IOException e) {
System.out.println("写入失败");
System.exit(1);
}
}
}
}).start();
} /**
* 使用PipeInputStream创建一个读取的线程
*/ public static void receive() {
new Thread(new Runnable() {
@Override
public void run() {
byte[] bytes = new byte[100]; //一次性只读取100个字节
int len = 0;
try {
len = inputStream.read(bytes, 0, 100); //读取
while (len != -1) {
System.out.println("已经读取了" + len + "个字节");
len = inputStream.read(bytes, 0, 100);
} } catch (IOException e) {
System.out.println("读取失败");
System.exit(1);
} }
}).start(); } public static void main(String args[]) {
try {
inputStream.connect(outputStream); //连接
} catch (IOException e) {
System.out.println("连接失败");
System.exit(1);
}
send();
receive(); } }

注意:从上面的运行结果可以看出,缓冲区最多可以写入1024个字节的数据,所以在缓冲区满了之后上面的send进程就会堵塞等待缓冲区空闲,如果recieve进程不继续读取数据了,那么就会一直出现堵塞

问题

  • 写线程正在往缓冲区写数据的时候,但是此时的读线程的结束读取,那么此时的写线程的管道就会发生IOException异常,可以将上面receive进程中的while(true)去掉就可以清楚的看出
  • 读线程正在从缓冲区读数据的时候,但是此时的写线程的管道已经结束了,此时就会引起读线程的管道发生IOException异常,将上面的send进程中的while(true)去掉就可以实现这个问题
  • 必须是启用多线程才能实现管道之间的读写,否则会出现堵塞现象,因为这里的PipeOutputStream每次向缓冲区写入的字节数最大是1024,如果不及时的减少缓冲区的数据量就会出现堵塞

解决方法

  • 后续更新中..........

参考文章

Java IO学习笔记五的更多相关文章

  1. Java IO学习笔记五:BIO到NIO

    作者:Grey 原文地址: Java IO学习笔记五:BIO到NIO 准备环境 准备一个CentOS7的Linux实例: 实例的IP: 192.168.205.138 我们这次实验的目的就是直观感受一 ...

  2. Java IO学习笔记六:NIO到多路复用

    作者:Grey 原文地址:Java IO学习笔记六:NIO到多路复用 虽然NIO性能上比BIO要好,参考:Java IO学习笔记五:BIO到NIO 但是NIO也有问题,NIO服务端的示例代码中往往会包 ...

  3. Java IO学习笔记:概念与原理

    Java IO学习笔记:概念与原理   一.概念   Java中对文件的操作是以流的方式进行的.流是Java内存中的一组有序数据序列.Java将数据从源(文件.内存.键盘.网络)读入到内存 中,形成了 ...

  4. Java IO学习笔记总结

    Java IO学习笔记总结 前言 前面的八篇文章详细的讲述了Java IO的操作方法,文章列表如下 基本的文件操作 字符流和字节流的操作 InputStreamReader和OutputStreamW ...

  5. Java IO学习笔记三

    Java IO学习笔记三 在整个IO包中,实际上就是分为字节流和字符流,但是除了这两个流之外,还存在了一组字节流-字符流的转换类. OutputStreamWriter:是Writer的子类,将输出的 ...

  6. Java IO学习笔记二

    Java IO学习笔记二 流的概念 在程序中所有的数据都是以流的方式进行传输或保存的,程序需要数据的时候要使用输入流读取数据,而当程序需要将一些数据保存起来的时候,就要使用输出流完成. 程序中的输入输 ...

  7. Java IO学习笔记一

    Java IO学习笔记一 File File是文件和目录路径名的抽象表示形式,总的来说就是java创建删除文件目录的一个类库,但是作用不仅仅于此,详细见官方文档 构造函数 File(File pare ...

  8. Java IO学习笔记一:为什么带Buffer的比不带Buffer的快

    作者:Grey 原文地址:Java IO学习笔记一:为什么带Buffer的比不带Buffer的快 Java中为什么BufferedReader,BufferedWriter要比FileReader 和 ...

  9. Java IO学习笔记二:DirectByteBuffer与HeapByteBuffer

    作者:Grey 原文地址:Java IO学习笔记二:DirectByteBuffer与HeapByteBuffer ByteBuffer.allocate()与ByteBuffer.allocateD ...

随机推荐

  1. 开始使用gentoo linux——gentoo安装笔记(下)

    gentoo安装笔记(下) 上一章,已经对操作系统安装做了充分准备,并且已经从livecd(u盘系统)切换进入了gentoo安装环境中. 不过现在才是真正的开始!打起精神!这可不是在装ubuntu! ...

  2. Maven的pom.xml配置文件详解

    Maven简述 Maven项目对象模型(POM),可以通过一小段描述信息来管理项目的构建,报告和文档的软件项目管理工具. Maven 除了以程序构建能力为特色之外,还提供高级项目管理工具.由于 Mav ...

  3. WebGIS开源解决方案之开发环境搭建(四)

    续前几篇文章,前面陆续介绍了开源GIS服务器Geoserver,开源数据库Postpresql以及开源前端udig的安装和基本使用. WebGIS前端开发,可以选择arcgis for javascr ...

  4. ListView在异步加载动态图片时,往往最后一项或几项被遮盖(IM场景居多)

    如果ListView中得默认图片比较小,新图片加载后,撑大ListView中的对应项,导致最后一项或几项被覆盖. 解决思路: 1.默认图片设定和新图大小一样,换句话说,新图加载后转成和默认图片一样的大 ...

  5. Ninja 之路:试炼!求生演习——异步 I/O、http

    鸣人火影之路的第一步,就是跟着卡卡西学习基本的忍术,让自己先在忍者的世界里生存下来,so,想要在 node 的世界里游刃有余,必须要掌握异步 I/O.http等核心技能. ok,第一步先学会读懂需求 ...

  6. 根据优先关系矩阵使用逐次加一发构造优先函数(Java)

    若已知运算符之间的优先关系,可按如下步骤构造优先函数: 1.对每个运算符a(包括#在内)令f(a)=g(a)=1 2.如果a⋗b且f(a)<=g(b)令f(a)=g(b)+1 3.如果a⋖b且f ...

  7. 【转】解决response.AddHeader("Content-Disposition", "attachment; fileName=" + fileName) 中文显示乱码

    如果fileName为中文则乱码.解决办法是 方法1: response.setHeader("Content-Disposition", "attachment; fi ...

  8. CountDownLatch——闭锁的实现之一

    CountDownLatch实际上是一种闭锁实现.闭锁:是一种同步工具类,可以延迟线程的进度知道其到达终止状态--<Java并发编程实战>.这个怎么解释呢?简单来说,就是有1个线程需要等待 ...

  9. JDFS:一款分布式文件管理实用程序第一篇(线程池、epoll、上传、下载)

    一 前言 截止目前,笔者在博客园上面已经发表了3篇关于网络下载的文章,这三篇博客实现了基于socket的http多线程远程断点下载实用程序.笔者打算在此基础上开发出一款分布式文件管理实用程序,截止目前 ...

  10. 15、TCP/IP协议

    15.TCP/IP协议       几台孤立计算机系统组在一起形成网络,几个孤立网络连在一起形成一个网络的网络,即互连网.一个互连网就是一组通过相同协议族互连在一起的网络. 互联网的目的之一是在应用程 ...