Java IO学习笔记五
管道流
- 管道流的主要作用是可以进行两个线程间的通讯,分为管道输出流(
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学习笔记五的更多相关文章
- Java IO学习笔记五:BIO到NIO
作者:Grey 原文地址: Java IO学习笔记五:BIO到NIO 准备环境 准备一个CentOS7的Linux实例: 实例的IP: 192.168.205.138 我们这次实验的目的就是直观感受一 ...
- Java IO学习笔记六:NIO到多路复用
作者:Grey 原文地址:Java IO学习笔记六:NIO到多路复用 虽然NIO性能上比BIO要好,参考:Java IO学习笔记五:BIO到NIO 但是NIO也有问题,NIO服务端的示例代码中往往会包 ...
- Java IO学习笔记:概念与原理
Java IO学习笔记:概念与原理 一.概念 Java中对文件的操作是以流的方式进行的.流是Java内存中的一组有序数据序列.Java将数据从源(文件.内存.键盘.网络)读入到内存 中,形成了 ...
- Java IO学习笔记总结
Java IO学习笔记总结 前言 前面的八篇文章详细的讲述了Java IO的操作方法,文章列表如下 基本的文件操作 字符流和字节流的操作 InputStreamReader和OutputStreamW ...
- Java IO学习笔记三
Java IO学习笔记三 在整个IO包中,实际上就是分为字节流和字符流,但是除了这两个流之外,还存在了一组字节流-字符流的转换类. OutputStreamWriter:是Writer的子类,将输出的 ...
- Java IO学习笔记二
Java IO学习笔记二 流的概念 在程序中所有的数据都是以流的方式进行传输或保存的,程序需要数据的时候要使用输入流读取数据,而当程序需要将一些数据保存起来的时候,就要使用输出流完成. 程序中的输入输 ...
- Java IO学习笔记一
Java IO学习笔记一 File File是文件和目录路径名的抽象表示形式,总的来说就是java创建删除文件目录的一个类库,但是作用不仅仅于此,详细见官方文档 构造函数 File(File pare ...
- Java IO学习笔记一:为什么带Buffer的比不带Buffer的快
作者:Grey 原文地址:Java IO学习笔记一:为什么带Buffer的比不带Buffer的快 Java中为什么BufferedReader,BufferedWriter要比FileReader 和 ...
- Java IO学习笔记二:DirectByteBuffer与HeapByteBuffer
作者:Grey 原文地址:Java IO学习笔记二:DirectByteBuffer与HeapByteBuffer ByteBuffer.allocate()与ByteBuffer.allocateD ...
随机推荐
- MongoDB副本集的搭建
副本集是mongodb提供的一种高可用解决方案.相对于原来的主从复制,副本集能自动感知primary节点的下线,并提升其中一个Secondary作为Primary. 整个过程对业务透明,同时也大大降低 ...
- iOS APP开发设置启动图片 Launch Image
一.添加启动图片 点击Assets.xcassets进入图片管理,右击,弹出"New Launch Image"或点下面的+号创建Launch Image: 这里首先说明一下尺寸: ...
- 原创-angularjs2不同组件间的通信
AngualrJs2官方方法是以@Input,@Output来实现组件间的相互传值,而且组件之间必须父子关系,下面给大家提供一个简单的方法,实现组件间的传值,不仅仅是父子组件,跨模块的组件也可以实现传 ...
- PHP自定义函数
啊哈
- 微软Build 2017第二天 .NET Standard 2.0 Preview 的客户端跨平台
微软公司一年一度的开发者大会,即“Microsoft Build 2017”在总部西雅图正式开幕.按照官方安排,本次大会将持续 3 天,主题围绕微软公司各项最新技术成果的展示和研讨,包括与微软相关的产 ...
- ASP.NET MVC5+EF6+EasyUI 后台管理系统(83)-Easyui Datagrid 行内编辑扩展
这次我们要从复杂的交互入手来说明一些用法,这才能让系统做出更加复杂的业务,上一节讲述了Datagird的批量编辑和提交本节主要演示扩展Datagrid行内编辑的属性,下面来看一个例子,我开启编辑行的时 ...
- Android GreenDAO3.0——介绍
引言 最近,学东西比较零散,各种知识混杂,于是记下学习记录,免得又忘了. 官方网址:http://greenrobot.org/greendao/documentation/introduction/ ...
- RabbitMQ集群和失败处理
RabbitMQ内建集群的设计用于完成两个目标:允许消费者和生产者在RabbitMQ节点在奔溃的情况下继续运行,以及通过添加更多的节点来线性扩展消息通信的吞吐量.当失去一个RabbitMQ节点时客户端 ...
- 广义后缀树(GST)算法的简介
导言 最近软件安全课上,讲病毒特征码的提取时,老师讲了一下GST算法.这里就做个小总结. 简介 基本信息 广义后缀树的英文为Generalized Suffix Tree,简称GST. 算法目的 ...
- 【wannacry病毒之暗网】-如何访问"暗网"(慎入)
心里能力不强的人,请别看. 有些事情还是不要接触比较好, 社会最恶一面不是随随便便就能接触到的, 也不是你能理解的 你想要用暗网做什么是你考虑的一个问题 什么是暗网? 所谓的"暗网" ...