【并发设计模式】聊聊等待唤醒机制的规范实现

在多线程编程中,其实就是分工、协作、互斥。在很多场景中,比如A执行的过程中需要同步等待另外一个线程处理的结果,这种方式下,就是一种等待唤醒的机制。本篇我们来讲述等待唤醒机制的三种实现,以及对应的应用场景。

Guarded Suspension 模式

Guarded Suspension 翻译过来就是保护性暂停。其实就是一个线程需要等待获取另外一个线程执行的结果,先把当前线程挂起,另外一个线程执行完毕之后,通知自己,结束阻塞状态,继续执行。

等待唤醒的规范实现如下:

  • sychronized+wait/notify/notifyAll
  • reentrantLock+Condition(await/singal/singalAll)
  • cas+park/unpark

其实底层以来的是pthread,pthread_mutex_lock/unlock pthread_cond_wait/singal。这里就不介绍了,感兴趣的朋友可以自行查阅。

解决线程之间的协作不可避免会用到阻塞唤醒机制

实际编码

syn

java 复制代码
package com.jia.suspension;

import java.util.Objects;
import java.util.concurrent.TimeUnit;

/**
 * @author qxlx
 * @date 2023/12/30 3:13 PM
 */
public class SynTest {

    private Object obj;

    public Object read() {
        synchronized (this) {
            while (Objects.isNull(obj)) {
                try {
                    System.out.println(Thread.currentThread().getName()+  " wait-before");
                    this.wait();
                    System.out.println(Thread.currentThread().getName()+  " wait-after");
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            }
            return obj;
        }
    }

    public void write() {
        System.out.println(Thread.currentThread().getName()+ "  write");
        synchronized (this) {
            obj = new Object();
            System.out.println(Thread.currentThread().getName()+ "  notifyAll-before");
            this.notifyAll();
            System.out.println(Thread.currentThread().getName()+ "   notifyAll-after");
        }
    }


    public static void main(String[] args) throws InterruptedException {
        SynTest synTest = new SynTest();

        new Thread(()-> {
            synTest.read();
        }).start();

        new Thread(()-> {
            synTest.write();
        }).start();

        TimeUnit.SECONDS.sleep(2);
    }

}

切记 不能在main线程中启动,需要单独创建两个线程去执行,否则main线程阻塞的话,程序就会阻塞不会执行下去。

conditon

java 复制代码
package com.jia.suspension;

import java.util.concurrent.TimeUnit;
import java.util.concurrent.locks.Condition;
import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReentrantLock;

/**
 * @author qxlx
 * @date 2023/12/30 3:31 PM
 */
public class ConditionTest {

    private final Lock lock = new ReentrantLock();
    private final Condition condition = lock.newCondition();
    private Object obj;

    public Object read () {
        try {
            lock.lock();
            while (obj == null) {
                System.out.println("getLock");
                condition.await();
            }
        } catch (Exception e) {
            e.printStackTrace();
        } finally {
            lock.unlock();
        }
        return obj;
    }

    public void write() {
        try {
            lock.lock();
            obj = new Object();
            condition.signalAll();
            System.out.println("唤醒");
        } catch (Exception e) {
            e.printStackTrace();
        } finally {
            lock.unlock();
        }
    }

    public static void main(String[] args) throws InterruptedException {
        ConditionTest test = new ConditionTest();

        new Thread(()-> {
            test.read();
        }).start();

        new Thread(()-> {
            test.write();
        }).start();

        TimeUnit.SECONDS.sleep(1);
    }


}

LockSouport

java 复制代码
package com.jia.suspension;

import java.util.concurrent.locks.LockSupport;

/**
 * @author qxlx
 * @date 2023/12/30 3:38 PM
 */
public class LockSupportTest {

    private Object obj;

    public Object read() {
        while (obj == null) {
            System.out.println("read-线程等待");
            LockSupport.park();
            System.out.println("read-线程唤醒");
        }
        return obj;
    }

    public void write(Thread thread) {
        obj = new Object();
        LockSupport.unpark(thread);
        System.out.println("唤醒线程");
    }

    public static void main(String[] args) {
        LockSupportTest lockSupportTest = new LockSupportTest();

        Thread thread = new Thread(() -> {
            lockSupportTest.read();
        });

        thread.start();

        Thread thread2 = new Thread(() -> {
            lockSupportTest.write(thread);
        });

        thread2.start();
    }

}

好了以上就是三种唤醒阻塞的方式。

应用场景

  • 多线程环境下多个线程访问相同实例资源,从实例资源中获得资源并处理;
  • 实例资源需要管理自身拥有的资源,并对请求线程的请求作出允许与否的判断

在实际的开发中,我们对外提供一个API数据查询的接口,但是需要以来下游系统进行组合数据,将结果写入MQ,下游服务处理完毕后,然后另外一个线程进行获取数据处理。

从图中可以看从处理web请求的是蓝色的线程,而从Topic获取数据的线程是红色线程,也就是蓝色线程异步写入Topic数据后,会阻塞,等待红色线程获取结果后,然后在返回结果。

相关推荐
微风中的麦穗2 小时前
【MATLAB】MATLAB R2025a 详细下载安装图文指南:下一代科学计算与工程仿真平台
开发语言·matlab·开发工具·工程仿真·matlab r2025a·matlab r2025·科学计算与工程仿真
2601_949146532 小时前
C语言语音通知API示例代码:基于标准C的语音接口开发与底层调用实践
c语言·开发语言
开源技术3 小时前
Python Pillow 优化,打开和保存速度最快提高14倍
开发语言·python·pillow
学嵌入式的小杨同学3 小时前
从零打造 Linux 终端 MP3 播放器!用 C 语言实现音乐自由
linux·c语言·开发语言·前端·vscode·ci/cd·vim
毕设源码-朱学姐3 小时前
【开题答辩全过程】以 基于JavaWeb的网上家具商城设计与实现为例,包含答辩的问题和答案
java
mftang4 小时前
Python 字符串拼接成字节详解
开发语言·python
jasligea5 小时前
构建个人智能助手
开发语言·python·自然语言处理
kokunka5 小时前
【源码+注释】纯C++小游戏开发之射击小球游戏
开发语言·c++·游戏
C雨后彩虹5 小时前
CAS与其他并发方案的对比及面试常见问题
java·面试·cas·同步·异步·
云栖梦泽5 小时前
易语言开发从入门到精通:补充篇·网络编程进阶+实用爬虫开发·API集成·代理IP配置·异步请求·防封禁优化
开发语言