多线程、高并发笔记(五):线程间的通信

等待/通知机制

  • 什么是等待通知机制?

    在单线程的编程中,要执行的操作需要满足一定的条件下才能执行,可以把这个操作放在if语句块中。

    在多线程编程中,可能A线程的条件,没有满足只是暂时的,稍后其他的线程B可能更新条件使得A线程的条件得到满足。可以将A线程暂停,直到它的条件得到满足后再将A线程唤醒。

  • 等待/通知机制的实现

    Object类中的wait()方法可以使执行当前代码的线程等待,暂停执行,直到接到通知或被中断为止。

//1.waitit()方法只能在同步代码块中由锁对象调用
//2.调用wait()方法,当前线程会释放锁
//伪代码
synchronized(锁对象){
    while(条件不成立){
        //通过锁对象调用wait()方法暂停线程,会释放锁对象
        锁对象.wait();
    }
    //线程的条件满足了继续向下执行
}

Object类的notify()可以唤醒线程,该方法也必须在同步代码块由锁对象调用,没有使用锁对象调用wait()
notify()会抛出IIegaIMonitorStateExeption异常。如果有多个等待的线程,notify()方法只能唤醒其中的一个。

在同步代码块中调用notify()后,并不会立即释放锁对象,需要等当前同步代码块执行完后才会释放锁对象,一般将notify()方法放在同步代码块的最后。

//伪代码
synchronized(所对象){
    //执行修改保护条件的代码
      //唤醒其他线程
    锁对象.notify();
}
  • notify()方法后不会立即释放锁对象
  • interrupt()方法会中断wait()

    当线程处于wait等待状态时,调用线程对象的interrupt()方法会中断线程的等待状态,会产生InterruptedException异常

  • notify()与notifyAll()

    notify()一次只能唤醒一个线程,如果有多个等待的线程,只能随机唤醒其中的一个;想要唤醒所有的等待线程,需要调用notifyAll().

  • wait(long)的使用

    wait(long)带有long类型参数的wait()等待,如果在参数指定的时间内没有被唤醒,超时后会自动唤醒。

  • 通知过早

    线程wait等待后可以调用notify唤醒线程,如果唤醒的过早,在等待之前就调用了,会打乱程序正常的逻辑。

  • wait等待的条件发生了变化

    在使用wait/notify模式时,注意wait条件发生了变化,也可能会造成程序逻辑的混乱

生产者消费者模式

在java中,负责产生数据的模块是生产者,负责使用数据的模块是消费者。解决数据的平衡问题,即先有数据才能使用,没有数
据时,消费者需要等待。

  • 生产-消费:操作值
  • 操作栈

    使生产者把数据存储到List集合中,消费者从List集合中取数据,使用List集合模拟栈。

通过管道实现线程间的通信

在java.io包中的PipeStream管道流用于在线程之间传送数据。
一个线程发送数据到输出管道,另外一个线程从输入管道中读取数据。
相关的类包括:PipedlnputStream和PipedOutputStream (字节流) 和 PipedWriter (字符流)。

ThreadLocal的使用

除了控制资源的访问之外,还可以通过增加资源来保证线程安全。ThreadLocal主要解决为每个线程绑定自己的值。

  • 使用ThreadLocal 为每个线程指定自己的对象
  • ThreadLocal 初始值 :(1).定义一个ThreadLocal的子类 (2).写initialValue方法,设置初始值 (这样第一次调用get()就不会返回null)
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值