27.Java 线程间通信(synchronized 实现线程间通信、Lock 实现线程间通信)

一、线程间通信

1、概述
  • 线程间通信的模型有两种:共享内存和消息传递
2、多线程编程步骤(中)
  1. 创建资源类,在资源类中创建属性和操作方法

  2. 在资源类操作方法进行判断、操作、通知

  3. 创建多个线程,调用资源类中的操作方法


二、synchronized 实现线程间通信

1、需求
  • 通过使用两个线程对值(0)进行操作,一个线程加 1,一个线程减 1,交替实现多次
2、具体实现
(1)资源类
  • Share 类
java 复制代码
package com.my.communicate;

public class Share {

    // 初始值
    private int number;

    // 加 1 的方法
    public synchronized void increase() throws InterruptedException {

        // number 为 0 则等待
        if (number != 0) {
            this.wait();
        }
        number++;
        System.out.println(Thread.currentThread().getName() + "  " + number);

        // 通知其他线程
        this.notifyAll();
    }

    // 减 1 的方法
    public synchronized void decrease() throws InterruptedException {

        // number 为 0 则等待
        if (number != 1) {
            this.wait();
        }
        number--;
        System.out.println(Thread.currentThread().getName() + "  " + number);

        // 通知其他线程
        this.notifyAll();
    }
}
(2)多线程测试
  • ShareTest 类
java 复制代码
package com.my.communicate;

public class ShareTest {
    public static void main(String[] args) {
        Share share = new Share();

        Thread thread1 = new Thread(() -> {
            for (int i = 0; i < 10; i++) {
                try {
                    share.increase();
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            }
        }, "AA");
        Thread thread2 = new Thread(() -> {
            for (int i = 0; i < 10; i++) {
                try {
                    share.decrease();
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            }
        }, "BB");

        thread1.start();
        thread2.start();
    }
}
3、虚假唤醒
  • 在当前 synchronized 实现线程间通信案例中再增加两个线程,执行结果会不符合预期,根本原因是虚假唤醒问题

  • 如果一个线程执行完毕后,通知其他线程,该线程又进入等待睡眠,被唤醒后,if 语句不会进行判断

  • 需要将 if 语句换成 while 语句,因为在哪里等待睡眠就会在哪里被唤醒,需要使用 while 语句再次进行判断

4、改进
(1)资源类
  • ShareImprove 类
java 复制代码
package com.my.communicate;

public class ShareImprove {

    // 初始值
    private int number;

    // 加 1 的方法
    public synchronized void increase() throws InterruptedException {

        // number 为 0 则等待
        while (number != 0) {
            this.wait();
        }
        number++;
        System.out.println(Thread.currentThread().getName() + "  " + number);

        // 通知其他线程
        this.notifyAll();
    }

    // 减 1 的方法
    public synchronized void decrease() throws InterruptedException {

        // number 为 0 则等待
        while (number != 1) {
            this.wait();
        }
        number--;
        System.out.println(Thread.currentThread().getName() + "  " + number);

        // 通知其他线程
        this.notifyAll();
    }
}
(2)多线程测试
  • ShareTestImprove 类
java 复制代码
package com.my.communicate;

public class ShareTestImprove {
    public static void main(String[] args) {
        Share share = new Share();

        Thread thread1 = new Thread(() -> {
            for (int i = 0; i < 10; i++) {
                try {
                    share.increase();
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            }
        }, "AA");
        Thread thread2 = new Thread(() -> {
            for (int i = 0; i < 10; i++) {
                try {
                    share.decrease();
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            }
        }, "BB");
        Thread thread3 = new Thread(() -> {
            for (int i = 0; i < 10; i++) {
                try {
                    share.increase();
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            }
        }, "CC");
        Thread thread4 = new Thread(() -> {
            for (int i = 0; i < 10; i++) {
                try {
                    share.decrease();
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            }
        }, "DD");

        thread1.start();
        thread2.start();
        thread3.start();
        thread4.start();
    }
}

三、Lock 实现线程间通信

1、需求
  • 通过使用两个线程对值(0)进行操作,一个线程加 1,一个线程减 1,交替实现多次
2、具体实现
  • LShare 类
(1)资源类
java 复制代码
package com.my.communicate;

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

public class LShare {

    // 初始值
    private int number;

    // 创建可重入锁
    private Lock lock;

    // 创建 Condition 对象
    private Condition condition;

    public LShare() {
        number = 0;
        lock = new ReentrantLock();
        condition = lock.newCondition();
    }

    // 加 1 的方法
    public void increase() {
        lock.lock();
        try {
            while (number != 0) {

                // 等待
                condition.await();
            }
            number++;
            System.out.println(Thread.currentThread().getName() + "  " + number);

            // 通知其他线程
            condition.signalAll();
        } catch (InterruptedException e) {
            e.printStackTrace();
        } finally {
            lock.unlock();
        }
    }

    // 减 1 的方法
    public void decrease() {
        lock.lock();
        try {
            while (number != 1) {

                // 等待
                condition.await();
            }
            number--;
            System.out.println(Thread.currentThread().getName() + "  " + number);

            // 通知其他线程
            condition.signalAll();
        } catch (InterruptedException e) {
            e.printStackTrace();
        } finally {
            lock.unlock();
        }
    }
}
(2)多线程测试
  • LShareTest 类
java 复制代码
package com.my.communicate;

public class LShareTest {
    public static void main(String[] args) {
        LShare lShare = new LShare();

        Thread thread1 = new Thread(() -> {
            for (int i = 0; i < 10; i++) {
                lShare.increase();
            }
        }, "AA");
        Thread thread2 = new Thread(() -> {
            for (int i = 0; i < 10; i++) {
                lShare.decrease();
            }
        }, "BB");
        Thread thread3 = new Thread(() -> {
            for (int i = 0; i < 10; i++) {
                lShare.increase();
            }
        }, "CC");
        Thread thread4 = new Thread(() -> {
            for (int i = 0; i < 10; i++) {
                lShare.decrease();
            }
        }, "DD");

        thread1.start();
        thread2.start();
        thread3.start();
        thread4.start();
    }
}
相关推荐
_r0bin_1 小时前
前端面试准备-7
开发语言·前端·javascript·fetch·跨域·class
zhang98800001 小时前
JavaScript 核心原理深度解析-不停留于表面的VUE等的使用!
开发语言·javascript·vue.js
恸流失1 小时前
DJango项目
后端·python·django
硅的褶皱2 小时前
对比分析LinkedBlockingQueue和SynchronousQueue
java·并发编程
MoFe12 小时前
【.net core】天地图坐标转换为高德地图坐标(WGS84 坐标转 GCJ02 坐标)
java·前端·.netcore
季鸢3 小时前
Java设计模式之观察者模式详解
java·观察者模式·设计模式
Fanxt_Ja3 小时前
【JVM】三色标记法原理
java·开发语言·jvm·算法
蓝婷儿3 小时前
6个月Python学习计划 Day 15 - 函数式编程、高阶函数、生成器/迭代器
开发语言·python·学习
love530love3 小时前
【笔记】在 MSYS2(MINGW64)中正确安装 Rust
运维·开发语言·人工智能·windows·笔记·python·rust
Mr Aokey4 小时前
Spring MVC参数绑定终极手册:单&多参/对象/集合/JSON/文件上传精讲
java·后端·spring