1. 概述:

    PipedReader 和 PipedWriter,意为管道读写流。所谓管道,那就是有进有出,所以这也是它们跟其它流对象最显著的区别:PipedReader和PipedWriter必须成对使用才有意义。

  2. 角色扮演:

    PipedWriter 扮演生产者的角色,将字符数据写入到管道;PipedReader扮演消费者的角色,负责将数据从管道取出消费掉。

  3. 数据结构:

    要实现一个管道,至少需要三个要素:一个线性存储结构、一个写入位置标记、一个读取位置标记。java中把这三个要素放在了PipedReader中:

    

    从代码注释可以看出,buffer[] 就是存储数据的线性结构、in 是写入的位置标记、out是读取的位置标记。"The circular buffer" 说明buffer数组是循环使用,那么肯定会牵涉到数据覆盖的问题,后文中会讲到怎么解决。注释中还说:in<0 表示管道为空(没有数据),in==out 表示管道满了。

  4. 构造:

    4.1. PipedWriter有两个构造方法,一个无参构造方法,一个带有一个PipedReader对象的构造方法。使用前者时只是单单构造一个Writer对象,没有建立连接,需要后续手动调用connect方法建立连接才能write数据;使用后者时会在构造完成时调用connect方法建立管道连接:

      

    4.2. PipedReader有4个构造方法,两个参数可选:需要建立连接的Writer对象和管道长度(默认1024):

      

      

    可以看出,PipedWriter和PipedReader都提供了带有连接对象参数的构造函数,所以实际应用中,他们之间构造的先后顺序可以自己决定。

  5. 建立连接:

    PipedReader中保存着连接标志: boolean connected = false; PipedWriter中保存着对PipedReader的引用: private PipedReader sink; 两个类中都有 connect 方法,而其实 PipedReader 中的 conncet 方法最终也是通过调用 PipedWriter 中的 connect 方法,如下:

    

    我们看到,连接方法做了以下几件事:

      5.1. 异常检查,包括:为空检查、已连接检查、已关闭检查(通过同步机制,保证这些检查的结果是可信的);

      5.2. 建立 PipedWriter 对 PipedReader 的引用;

      5.3. 初始化管道位置标记;

      5.4. 将连接标志置为 true;

  6. 写入数据:

    PipedWriter 有两个 write 方法,最终都是调用 PipedReader 的 receive 方法实现的(其实这很自然,因为 buffer 本身就是放在PipedReader中的),所以写入数据的核心方法是  synchronized void receive(int c) throws IOException :

      

    从代码可以看出:

      6.1. 写入方法是同步的。如果不同步,那么可能导致不同线程写入的数据相互覆盖;

     6.2. 写入方法是阻塞的。当管道已满却要写入数据的时候,会首先唤醒监听当前对象锁的所有线程(当然也包括读取线程)让他们进入就绪态,准备竞争锁资源;然后让当前执行写入的线程放弃对象锁,沉睡1秒。这样一来,才有可能让该对象锁上的读取线程从管道读取数据,将管道腾出部分或者全部的空间,当前线程才能继续写入数据,结束阻塞状态。

  7. 读取数据:

    PipedReader有两个 read 方法,分别是读取单个字符和多个,大同小异,下面是读取单个字符的方法:

      

      

    从代码可以看出:

      7.1. read 方法是同步的。

      7.2. read 方法是阻塞的。当管道已空却需要读取数据时,首先唤醒当前对象锁上的其它线程(包括写入线程);然后让当前执行读取的线程沉睡1秒,放弃锁资源,这样一来才有可能让该对象锁上的写入线程写入数据,当前线程才有数据可读,结束阻塞状态。

  8. 写入结束和关闭管道:

    可以从Writer或者Reader任意一方关闭管道, closeByReader 和 closeByWriter 会标记从哪一方关闭的,并且若是从 Writer 方关闭的,会调用 PipedReader 的  synchronized void receivedLast() 方法,从而唤醒所有线程。

  9. 总结:

    9.1. PipedReader 和 PipedWriter 需要成对使用,才能建立管道;

    9.2. 管道是通过一个循环使用的字符数组实现的;

    9.3. write 和 read 都是同步的;

    9.4. write 和 read 都是阻塞的(write 时管道已满就阻塞,read时管道已空就阻塞);

  10. 小小实践:

    https://www.cnblogs.com/coding-one/p/11352217.html

javaIO——PipedReader & PipedWriter的更多相关文章

  1. javaIO——PipedReader 和 PipedWriter 实现模拟即时聊天

    上一篇学习了javaIO里面的 PipedReader 和 PipedWriter,这里对两个类做一个小小的练习:实现一个即时消息发送和接收的聊天系统(这里只实现单向发送,双向同理,定义两个管道即可) ...

  2. Java 管道PipedInputStream PipedOutStream PipedReader PipedWriter

    java中的管道流(pipeStream)是一种特殊的流,用于在不同线程间直接传送数据.一个线程发送数据到输出管道,另外一个线程从输入管道中读取数据.通过使用管道,实现不同线程间的通信,而不必借助类似 ...

  3. Java-IO之PipedReader和PipedWriter

    PipedReader和PipedWriter与PipedInputStream和PipedOutputStream一样,都可以用于管道通信.PipedWriter是字符管道输出流,继承于Writer ...

  4. [十九]JavaIO之PipedReader 和 PipedWriter

    功能简介 还记得PipedInputStream  和 PipedOutputStream么 我们之前是这么说的: p, li { white-space: pre-wrap; } 使用管道通信时,必 ...

  5. Java_io体系之PipedWriter、PipedReader简介、走进源码及示例——14

    Java_io体系之PipedWriter.PipedReader简介.走进源码及示例——14 ——管道字符输出流.必须建立在管道输入流之上.所以先介绍管道字符输出流.可以先看示例或者总结.总结写的有 ...

  6. java io系列20之 PipedReader和PipedWriter

    本章,我们学习PipedReader和PipedWriter.它们和“PipedInputStream和PipedOutputStream”一样,都可以用于管道通信. PipedWriter 是字符管 ...

  7. Java学习日记之 Java-IO流

    Java中的IO流在处理上分为字节流和字符流.字节流和字符流的区别 : 1.字节流读取的时候,读到一个字节就返回一个字节:  字符流使用了字节流读到一个或多个字节(中文对应的字节数是两个,在UTF-8 ...

  8. [三]JavaIO之IO体系类整体设计思路 流的概念以及四大基础分类

    从本文开始,将正式进入JavaIO的简介 在继续javaIO系列的文章之前 可以过去看一下 本人博客上的设计模式中的 适配器模式和装饰器模式 这会对接下来的阅读大有帮助   本文是从逻辑上介绍整个的J ...

  9. [四] JavaIO之类层次体系结构横向比对

      IO家族类层次体系结构横向匹配   上一篇文章中主要介绍了JavaIO流家族的整体设计思路,简单回顾下 基本逻辑涉及数据源 流的方向,以及流的数据形式这三个部分的组合 按照流的数据形式和流的方向, ...

随机推荐

  1. https证书制作及springboot配置https

    1.生成秘钥 openssl genrsa -out private.key 2048 2.生成用于申请请求的证书文件csr,一般会将该文件发送给CA机构进行认证,本例使用自签名证书 openssl ...

  2. 数据分析 - Power BI

    BI 目的 单表的展示有限很多的时候只能体现现象, 仅仅进行监控级别没问题 但是就数据分析而言实在不够用, 大部分的事情需要多表多图展示才可以通过现象深入挖掘诱因 BI 安装 这里使用 microso ...

  3. python 时间对比

    import datetimed1 = datetime.datetime.strptime('2015-03-05  17:41:20', '%Y-%m-%d %H:%M:%S')d2 = date ...

  4. Linux云服务器磁盘不见了?解决方案在这里,云服务器磁盘挂载

    用过诸多种云以后,发现有个通病,就是新买的数据盘在机器中找不到.本篇总结一下此类问题的解决方法,望各位点赞,有问题评论区见 一.云服务和物理机一样,你买了云服务器的数据盘以后,就相当于把数据盘直接安装 ...

  5. 比特币nBits计算

    转载:比特币源码分析(二十二) - 挖矿和共识 https://blog.csdn.net/yzpbright/article/details/81231351 CalculateNextWorkRe ...

  6. 使用XCode7打包动态库(Framework)

    iOS中的静态库和动态库 概念 静态库(Static Library)以 .a 为后缀,它是你的源码的实现.m文件编译而成的二进制文件集合,需要配合上暴漏的.h文件使用,它在引用链接时拷贝至可执行文件 ...

  7. 查看进程的命令ps

    查看进程的命令:ps aux strace -p pid(进程id) 杀死进程:kill pid(进程id)强制杀死进程:kill -9 pid(进程id) linux ps 命令查看进程状态linu ...

  8. iOS-上传头像的使用

    static NSString *const uploadSuccess = @"更改头像成功"; @interface DMAccountInformationViewContr ...

  9. Nginx使用默认配置启动异常处理

    Ps1:错误问题:nginx: [error] OpenEvent("Global\ngx_reload_5988") failed (2: The system cannot f ...

  10. Linux生成ssh密钥免密登录,允许/禁止密码登录,允许/禁止root远程登录,更改ssh端口

    进入想要使用密钥登录的用户家目录 cd 或 cd ~ 执行密钥创建命令,不行可能需要下载openssh-server与openssh-client ssh-keygen -t rsa -P " ...