Java IO - PipedOutputStream & PipedInputStream

基本概念

  • PipedOutputStream & PipedInputStream (管道字节输入流&管道输出字节流)是配套使用的。可以将管道输出流连接到管道输入流来创建通信管道。

  • 通常的用法是让两个线程分别操作输入输出流,由输出流向管道写入数据,然后再由输出流从管道中读取数据。不建议对这两个对象尝试使用单个线程,因为这样可能会造成该线程死锁

  • 继承关系:

这里写图片描述

这里写图片描述


实例探究

1.生产者&消费者

  • 定义两个线程代表生产者/消费者,然后将管道的输出/输入流进行连接,观察输出结果。

  • 生产者(Producer ):即管道输出流(PipedOutputStream ),不断地向输入流输出数据。

  • 消费者(Consumer ):即管道输入流(PipedInputStream ),不断地从输出流读取数据。

//生产者线程,负责向输入流输出数据
class Producer extends Thread {

    private PipedOutputStream pos;

    public Producer(PipedOutputStream pos, String name) {
        super(name);
        this.pos = pos;
    }

    public void run() {
        int i = 0;
        try {
            while (true) {
                Thread.sleep(500);
                pos.write(i);
                i++;
            }
        } catch (Exception e) {
            e.printStackTrace();
        }
    }
}

//消费者线程,负责从输出流读取数据
class Consumer extends Thread {
    private PipedInputStream pis;

    public Consumer(PipedInputStream pis, String name) {
        super(name);
        this.pis = pis;
    }

    public void run() {
        try {
            while (true) {
                System.out.println(this.getName() + ":" + pis.read());
            }
        } catch (IOException e) {
            e.printStackTrace();
        }
    }
}

public class Test {
   
    public static void main(String[] args) throws IOException {
        PipedOutputStream pos = new PipedOutputStream();
        PipedInputStream pis = new PipedInputStream();

        //连接管道
        pos.connect(pis);
        Producer pro = new Producer(pos, "Producer");
        Consumer con = new Consumer(pis, "Consumer");

        //启动线程
        pro.start();
        con.start();
    }
}

// 输出结果:
// Consumer:0
// Consumer:1
// Consumer:2
// Consumer:3
// Consumer:4
// Consumer:5

2.管道流限制性

  • 管道流只能实现单向发送,如果要两个线程之间互通讯,则需要两个管道流

  • 观察输出结果,发现管道输出流每次只能与一个管道输入流通信,因此一个管道流只能用于两个线程间的通讯。不仅仅是管道流,其他 IO 方式都是一对一传输。

//采用上面的例子,省略相同代码...
public static void main(String[] args) throws IOException {
    PipedOutputStream pos = new PipedOutputStream();
    PipedInputStream pis = new PipedInputStream();
    pos.connect(pis);
    Producer pro = new Producer(pos,"pro");

    //定义了两个消费者线程(即两个管道输入流)
    Consumer c1 = new Consumer(pis,"c1");
    Consumer c2 = new Consumer(pis,"c2");
    pro.start();
    c1.start();
    c2.start();
}

// 输出结果:
// c2:0
// c1:1
// c2:2
// c1:3
// c2:4
// c1:5

源码探究

2.PipedInputStream

类结构图

这里写图片描述

成员变量


                
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

oxf

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值