Java——多线程高并发系列之线程间的通信(synchronized、Lock、Condition)

简介: Java——多线程高并发系列之线程间的通信(synchronized、Lock、Condition)

文章目录:


写在前面

Demo1(多线程通信之计数案例:synchronized实现方式)

Demo2(多线程通信之计数案例:Lock + Condition实现方式)

Demo3(多线程定制化通信之循环交替打印案例)


写在前面


关键字 synchronized 与 wait()/notify()这两个方法一起使用可以实现等待/通知模式。Lock 锁的 newContition()方法返回 Condition 对象,Condition 类也可以实现等待/通知模式。

使用Condition 类可以进行选择性通知. Condition 接口中比较常用的两个方法:await()会使当前线程等待,同时会释放锁,当其他线程调用 signal()时,线程会重新获得锁并继续执行,signal()用于唤醒一个等待的线程。


注意:在调用Condition的await()/signal()方法前,也需要线程持有相关的 Lock 锁。调用 await()后线程会释放这个锁,在 singal()调用后会从当前 Condition 对象的等待队列中,唤醒 一个线程,唤醒 的线程尝试获得锁, 一旦获得锁成功就继续执行。


也就是说,Condition接口中的await()方法就相当于之前与synchronized联合使用的Object类中的wait()方法;signal()方法就相当于之前与synchronized联合使用的Object类中的notify()方法。


Demo1(多线程通信之计数案例:synchronized实现方式)


这个案例主要实现让多个线程交替打印 0、1 这两个数字。incr 方法实现的增加1,当num不为0(此时为1)则进行wait等待,其余情况增加1,之后notifyAll唤醒其他等待的线程,这里使用while则是为了避免虚假唤醒的情况。


package test.count;
/**
 * 多线程计数案例:synchronized实现方式
 */
class Share {
    private int num = 0;
    public synchronized void incr() throws InterruptedException {
        while (num != 0) {
            this.wait();
        }
        num++;
        System.out.println(Thread.currentThread().getName() + " :: " + num);
        this.notifyAll();
    }
    public synchronized void decr() throws InterruptedException {
        while (num != 1) {
            this.wait();
        }
        num--;
        System.out.println(Thread.currentThread().getName() + " :: " + num);
        this.notifyAll();
    }
}
public class CountNum {
    public static void main(String[] args) {
        Share share = new Share();
        new Thread(new Runnable() {
            @Override
            public void run() {
                for (int i = 0; i < 10; i++) {
                    try {
                        share.incr();
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                }
            }
        },"线程A").start();
        new Thread(new Runnable() {
            @Override
            public void run() {
                for (int i = 0; i < 10; i++) {
                    try {
                        share.decr();
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                }
            }
        },"线程B").start();
        new Thread(new Runnable() {
            @Override
            public void run() {
                for (int i = 0; i < 10; i++) {
                    try {
                        share.incr();
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                }
            }
        },"线程C").start();
        new Thread(new Runnable() {
            @Override
            public void run() {
                for (int i = 0; i < 10; i++) {
                    try {
                        share.decr();
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                }
            }
        },"线程D").start();
    }
}



Demo2(多线程通信之计数案例:Lock + Condition实现方式)


与Demo1是一样的理解,只是这里实现方式变了。这里也是首先 lock.lock() 获取锁,incr 方法实现的增加1,当num不为0(此时为1)则进行await等待,其余情况增加1,之后signalAll唤醒其他等待的线程,这里使用while则是为了避免虚假唤醒的情况。最后为了确保锁一定能释放,将 lock.unlock() 写在finally子句中。


package test.count;
import java.util.concurrent.locks.Condition;
import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReentrantLock;
/**
 * 多线程计数案例:Lock实现方式
 */
class Share2 {
    private int num = 0;
    private Lock lock = new ReentrantLock();
    private Condition condition = lock.newCondition();
    public void incr() {
        lock.lock();
        try {
            while (num != 0) {
                condition.await();
            }
            num++;
            System.out.println(Thread.currentThread().getName() + " :: " + num);
            condition.signalAll();
        } catch (InterruptedException e) {
            e.printStackTrace();
        } finally {
            lock.unlock();
        }
    }
    public void decr() {
        lock.lock();
        try {
            while (num != 1) {
                condition.await();
            }
            num--;
            System.out.println(Thread.currentThread().getName() + " :: " + num);
        } catch (InterruptedException e) {
            e.printStackTrace();
        } finally {
            lock.unlock();
        }
    }
}
public class CountNum2 {
    public static void main(String[] args) {
        Share2 share2 = new Share2();
        new Thread(new Runnable() {
            @Override
            public void run() {
                for (int i = 0; i < 10; i++) {
                    share2.incr();
                }
            }
        },"线程A").start();
        new Thread(new Runnable() {
            @Override
            public void run() {
                for (int i = 0; i < 10; i++) {
                    share2.decr();
                }
            }
        },"线程B").start();
        new Thread(new Runnable() {
            @Override
            public void run() {
                for (int i = 0; i < 10; i++) {
                    share2.incr();
                }
            }
        },"线程C").start();
        new Thread(new Runnable() {
            @Override
            public void run() {
                for (int i = 0; i < 10; i++) {
                    share2.decr();
                }
            }
        },"线程D").start();
    }
}



Demo3(多线程定制化通信之循环交替打印案例)


这个案例实现的是:三个线程(a、b、c)循环交替打印,也就是线程a打印5行、线程b接着打印10行、线程c继续打印15行、线程a再打印5行......,这样一直循环下去。


首先是定义了一个标志位 flag(线程a对应1、线程b对应2、线程c对应3),三个方法中依次获取锁,然后满足条件就打印、不满足就等待,之后每打印完一次就修改下标志位为下一个打印的线程标志位,之后将要打印的线程唤醒即可。


package test;
import java.util.concurrent.locks.Condition;
import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReentrantLock;
/**
 * 线程间的定制化通信
 */
class ShareResource {
    //定义标志位
    //1 ---> a , 2 ---> b , 3 ---> c
    private int flag = 1;
    private Lock lock = new ReentrantLock();
    private Condition a = lock.newCondition(); // a
    private Condition b = lock.newCondition(); // b
    private Condition c = lock.newCondition(); // c
    //打印5次,参数为打印的轮数
    public void print5(int loop) throws InterruptedException {
        lock.lock();
        try {
            while (flag != 1) {
                a.await();
            }
            for (int i = 1; i <= 5; i++) {
                System.out.println(Thread.currentThread().getName() + " :: " + i + " , 轮数: " + loop);
            }
            flag = 2;
            b.signal();
        } finally {
            lock.unlock();
        }
    }
    //打印10次,参数为打印的轮数
    public void print10(int loop) throws InterruptedException {
        lock.lock();
        try {
            while (flag != 2) {
                b.await();
            }
            for (int i = 1; i <= 10; i++) {
                System.out.println(Thread.currentThread().getName() + " :: " + i + " , 轮数: " + loop);
            }
            flag = 3;
            c.signal();
        } finally {
            lock.unlock();
        }
    }
    //打印15次,参数为打印的轮数
    public void print15(int loop) throws InterruptedException {
        lock.lock();
        try {
            while (flag != 3) {
                c.await();
            }
            for (int i = 1; i <= 15; i++) {
                System.out.println(Thread.currentThread().getName() + " :: " + i + " , 轮数: " + loop);
            }
            flag = 1;
            a.signal();
        } finally {
            lock.unlock();
        }
    }
}
public class ThreadDemo1 {
    public static void main(String[] args) {
        ShareResource shareResource = new ShareResource();
        new Thread(new Runnable() {
            @Override
            public void run() {
                for (int i = 1; i <= 3; i++) {
                    try {
                        shareResource.print5(i);
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                }
            }
        },"线程a").start();
        new Thread(new Runnable() {
            @Override
            public void run() {
                for (int i = 1; i <= 3; i++) {
                    try {
                        shareResource.print10(i);
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                }
            }
        },"线程b").start();
        new Thread(new Runnable() {
            @Override
            public void run() {
                for (int i = 1; i <= 3; i++) {
                    try {
                        shareResource.print15(i);
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                }
            }
        },"线程c").start();
    }
}

目录
打赏
0
0
0
0
85
分享
相关文章
【原理】【Java并发】【synchronized】适合中学者体质的synchronized原理
本文深入解析了Java中`synchronized`关键字的底层原理,从代码块与方法修饰的区别到锁升级机制,内容详尽。通过`monitorenter`和`monitorexit`指令,阐述了`synchronized`实现原子性、有序性和可见性的原理。同时,详细分析了锁升级流程:无锁 → 偏向锁 → 轻量级锁 → 重量级锁,结合对象头`MarkWord`的变化,揭示JVM优化锁性能的策略。此外,还探讨了Monitor的内部结构及线程竞争锁的过程,并介绍了锁消除与锁粗化等优化手段。最后,结合实际案例,帮助读者全面理解`synchronized`在并发编程中的作用与细节。
51 8
【原理】【Java并发】【synchronized】适合中学者体质的synchronized原理
|
20天前
|
【Java并发】【synchronized】适合初学者体质入门的synchronized
欢迎来到我的Java线程同步入门指南!我不是外包员工,梦想是写高端CRUD。2025年我正在沉淀中,博客更新速度加快,欢迎点赞、收藏、关注。 本文介绍Java中的`synchronized`关键字,适合初学者。`synchronized`用于确保多个线程访问共享资源时不会发生冲突,避免竞态条件、保证内存可见性、防止原子性破坏及协调多线程有序访问。
53 8
【Java并发】【synchronized】适合初学者体质入门的synchronized
Java网络编程,多线程,IO流综合小项目一一ChatBoxes
**项目介绍**:本项目实现了一个基于TCP协议的C/S架构控制台聊天室,支持局域网内多客户端同时聊天。用户需注册并登录,用户名唯一,密码格式为字母开头加纯数字。登录后可实时聊天,服务端负责验证用户信息并转发消息。 **项目亮点**: - **C/S架构**:客户端与服务端通过TCP连接通信。 - **多线程**:采用多线程处理多个客户端的并发请求,确保实时交互。 - **IO流**:使用BufferedReader和BufferedWriter进行数据传输,确保高效稳定的通信。 - **线程安全**:通过同步代码块和锁机制保证共享数据的安全性。
73 23
【JAVA】封装多线程原理
Java 中的多线程封装旨在简化使用、提高安全性和增强可维护性。通过抽象和隐藏底层细节,提供简洁接口。常见封装方式包括基于 Runnable 和 Callable 接口的任务封装,以及线程池的封装。Runnable 适用于无返回值任务,Callable 支持有返回值任务。线程池(如 ExecutorService)则用于管理和复用线程,减少性能开销。示例代码展示了如何实现这些封装,使多线程编程更加高效和安全。
Java 多线程 面试题
Java 多线程 相关基础面试题
Java多线程优化:提高线程池性能的技巧与实践
Java多线程优化:提高线程池性能的技巧与实践
190 1
Java多线程基础-11:工厂模式及代码案例之线程池(一)
本文介绍了Java并发框架中的线程池工具,特别是`java.util.concurrent`包中的`Executors`和`ThreadPoolExecutor`类。线程池通过预先创建并管理一组线程,可以提高多线程任务的效率和响应速度,减少线程创建和销毁的开销。
318 2
|
10月前
|
【Java多线程】对线程池的理解并模拟实现线程池
【Java多线程】对线程池的理解并模拟实现线程池
102 1
17 Java多线程(线程创建+线程状态+线程安全+死锁+线程池+Lock接口+线程安全集合)(下)
17 Java多线程(线程创建+线程状态+线程安全+死锁+线程池+Lock接口+线程安全集合)
115 6
17 Java多线程(线程创建+线程状态+线程安全+死锁+线程池+Lock接口+线程安全集合)(中)
17 Java多线程(线程创建+线程状态+线程安全+死锁+线程池+Lock接口+线程安全集合)
123 5
AI助理

你好,我是AI助理

可以解答问题、推荐解决方案等