线程间通信管道流

不念不忘少年蓝@ 2022-09-11 01:19 250阅读 0赞

管道输入/输出流

管道流是是一种使用比较少的线程间通信方式,管道输入/输出流和普通文件输入/输出流或者网络输出/输出流不同之处在于,它主要用于线程之间的数据传输,传输的媒介为管道。

管道输入/输出流主要包括4种具体的实现:PipedOutputStrean、PipedInputStrean、PipedReader和PipedWriter,前两种面向字节,后两种面向字符。

java的管道的输入和输出实际上使用的是一个循环缓冲数组来实现的,默认为1024,输入流从这个数组中读取数据,输出流从这个数组中写入数据,当这个缓冲数组已满的时候,输出流所在的线程就会被阻塞,当向这个缓冲数组为空时,输入流所在的线程就会被阻塞。

管道是一个进程间通信的概念,在要通信的进程间构建一个单向的数据流动的通道。数据通过该通道从一个进程流向另一个进程时是具有时间先后顺序的。就像是在进程间架起了一个”管道”。

image-20210910145315665

在java的类库中有2个管道类:PipedWriter(允许任务向管道写),和PipedReader(允许不同任务从同一管道中读取)。管道也可以理解为一个缓冲区,将要读写的内容存入到管道,输入输出都要从这个管道去操作,管道提供了一个封装好的解决方案。

image-20210910145329799

  1. package com.dongguo.sync;
  2. import java.io.IOException;
  3. import java.io.PipedReader;
  4. import java.io.PipedWriter;
  5. import java.util.concurrent.ExecutorService;
  6. import java.util.concurrent.Executors;
  7. import java.util.concurrent.TimeUnit;
  8. /** * @author Dongguo * @date 2021/8/24 0024-8:18 * @description: 任务间使用管道进行输入输出 */
  9. public class ThreadDemo6 {
  10. public static void main(String[] args) throws Exception {
  11. Sender sender = new Sender();
  12. Receiver receiver = new Receiver(sender);
  13. ExecutorService exec = Executors.newCachedThreadPool();
  14. exec.execute(sender);
  15. exec.execute(receiver);
  16. //休眠4秒钟后中断
  17. TimeUnit.SECONDS.sleep(4);
  18. exec.shutdownNow();
  19. }
  20. }
  21. class Sender implements Runnable {
  22. private PipedWriter out = new PipedWriter();
  23. public PipedWriter getPipedWriter() {
  24. return out;
  25. }
  26. public void run() {
  27. try {
  28. while (true) {
  29. for (char c = 'A'; c < 'Z'; c++) {
  30. out.write(c);
  31. //随机休眠500以内毫秒
  32. TimeUnit.MILLISECONDS.sleep(500);
  33. }
  34. }
  35. } catch (IOException e) {
  36. System.out.println(e + " Sender write exception");
  37. } catch (InterruptedException e) {
  38. System.out.println(e + " Sender sleep interrupt");
  39. }
  40. }
  41. }
  42. class Receiver implements Runnable {
  43. private PipedReader in;
  44. public Receiver(Sender sender) throws IOException {
  45. in = new PipedReader(sender.getPipedWriter());
  46. }
  47. public void run() {
  48. try {
  49. while (true) {
  50. System.out.println("Read: " + (char) in.read() + ",");
  51. }
  52. } catch (IOException e) {
  53. System.out.println(e + " Receiver read exception");
  54. }
  55. }
  56. }
  57. 运行结果:
  58. Read: A,
  59. Read: B,
  60. Read: C,
  61. Read: D,
  62. Read: E,
  63. Read: F,
  64. java.io.InterruptedIOException Receiver read exception
  65. java.lang.InterruptedException: sleep interrupted Sender sleep interrupt

Sender和Receiver分别代表了2个需要互相通信的任务(相当于生产者和消费者的角色),Sender 简历一个单独的PipedWriter对象,将信息写入到管道当中,而Receiver建立的PipedReader对象必须绑定到与Sender相关联的管道中,才能保证正确读取到数据。

发表评论

表情:
评论列表 (有 0 条评论,250人围观)

还没有评论,来说两句吧...

相关阅读

    相关 线通信管道

    管道输入/输出流 管道流是是一种使用比较少的线程间通信方式,管道输入/输出流和普通文件输入/输出流或者网络输出/输出流不同之处在于,它主要用于线程之间的数据传输,传输的媒介为

    相关 线通信

    为什么需要线程通讯 线程是操作系统调度的最小单位,有自己的栈空间,可以按照既定的代码逐步的执行,但是如果每个线程间都孤立的运行,那就会造资源浪费。所以在现实中,我们需要这