随笔 - 465  文章 - 0  评论 - 7  阅读 - 27万

廖雪峰Java11多线程编程-3高级concurrent包-3Condition

1.Condition实现等待和唤醒线程

java.util.locks.ReentrantLock用于替代synchronized加锁

但是synchronized可以使用wait和notify实现线程在条件不满足时的等待,条件满足时的唤醒。
如何使用ReentrantLock实现wait/notify的功能?
这个时候我们可以使用Condition对象来实现条件不满足时的等待,条件满足时的唤醒。

class TaskQueue{
    final Lock lock = new ReentrantLock()
    final Condition notEmpty = lock.newCondition();
}

1.1 对Condition对象调用await()方法就相当于对所对象调用wait()方法。

### 1.2 对Condition对象调用signalAll()就相当于在synchronized加锁对象调用notifyAll()方法。 Condition.await/signal/signalAll原理和wait/notify/notifyAll一致 * await()会释放当前锁,进入等待状态 * signal()会唤醒某个等待线程 * signalAll()会唤醒所有等待线程 * 唤醒线程从await()返回后需要重新获得锁

2. 示例

import java.util.LinkedList;
import java.util.Queue;
import java.util.concurrent.locks.Condition;
import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReentrantLock;

class TaskQueue{
    final Queue<String> queue = new LinkedList<>();
    final Lock lock = new ReentrantLock();
    final Condition notEmpty = lock.newCondition();
    public String getTask() throws InterruptedException{
        lock.lock();
        try{
            while(this.queue.isEmpty()){
                notEmpty.await();
            }
            return queue.remove();
        }finally {
            lock.unlock();
        }
    }
    public void addTask(String name){
        lock.lock();
        try{
            this.queue.add(name);
            notEmpty.signalAll();
        }finally {
            lock.unlock();
        }
    }
}
class WorkerThread extends Thread{
    TaskQueue taskQueue;
    public WorkerThread(TaskQueue taskQueue){
        this.taskQueue = taskQueue;
    }
    public void run(){
        while(!isInterrupted()){
            String name;
            try{
                name = taskQueue.getTask();
            }catch (InterruptedException e){
                break;
            }
            String result = "Hello, "+name+" !";
            System.out.println(result);
        }
    }
}
public class Main{
    public static void main(String[] args) throws InterruptedException{
        TaskQueue taskQueue = new TaskQueue();
        WorkerThread worker = new WorkerThread(taskQueue);
        worker.start();
        taskQueue.addTask("Bob");
        Thread.sleep(1000);
        taskQueue.addTask("Alice");
        Thread.sleep(1000);
        taskQueue.addTask("Tim");
        Thread.sleep(1000);
        worker.interrupt();
        worker.join();
        System.out.println("END");
    }
}
## 3. 总结 使用ReentrantLock+Condition可以更安全的实现线程的同步、等待和唤醒。 * Condition可以替代wait/notify * Condition对象必须从ReentrantLock对象获取 * ReentrantLock+Condition可以替代synchronized+wait/notify
posted on   singleSpace  阅读(200)  评论(0编辑  收藏  举报
编辑推荐:
· 深入理解 Mybatis 分库分表执行原理
· 如何打造一个高并发系统?
· .NET Core GC压缩(compact_phase)底层原理浅谈
· 现代计算机视觉入门之:什么是图片特征编码
· .NET 9 new features-C#13新的锁类型和语义
阅读排行:
· Spring AI + Ollama 实现 deepseek-r1 的API服务和调用
· 《HelloGitHub》第 106 期
· 数据库服务器 SQL Server 版本升级公告
· 深入理解Mybatis分库分表执行原理
· 使用 Dify + LLM 构建精确任务处理应用
< 2025年1月 >
29 30 31 1 2 3 4
5 6 7 8 9 10 11
12 13 14 15 16 17 18
19 20 21 22 23 24 25
26 27 28 29 30 31 1
2 3 4 5 6 7 8

点击右上角即可分享
微信分享提示