wait 和notify ,notifyAll,sleep

wait

使线程进入阻塞状态,释放CPU,以及锁

sleep

使线程进入睡眠状态,sleep方法不会释放CPU资源和锁资源,而是让出CPU的使用权。操作系统会将CPU分配给其他就绪线程,但当前线程依然存在,不会释放其占用的内存或其他资源。当调用Thread.sleep(milliseconds)时,当前线程会从"运行状态(RUNNABLE)"进入"计时等待状态(TIMED_WAITING)"。线程并没有被销毁,只是暂时停止执行,等待指定时间后重新进入就绪队列,等待CPU调度

notify

随机唤醒一个阻塞的线程

notifyAll

唤醒全部阻塞的线程

实现一个方法:两个线程交替打印奇数和偶数到100,线程A打印奇数,线程B打印偶数,交替打印

两个线程同时开始抢锁,如果是奇数线程抢到了锁,奇数线程判断当前是不是奇数,偶数线程判断当前是不是偶数

  • 第一次:如果偶数线程抢到了锁,但是不是偶数,就直接当前线程wait,释放锁,让奇数线程获取到锁,
  • 奇数执行打印数字,并给数字加1,然后唤醒其他线程(偶数数线程),此时奇数线程还拥有着锁,于是在此判断当前线
    数字是不是奇数,不是就执行wait方法,释放锁。
  • 此时偶数线程就可以获取到锁,然后执行完之后,数字加一,然后唤醒奇数线程,
  • 此时同样的,偶数线程还有锁,于是再次进入方法,判断是不是偶数,不是,就执行wait方法,释放锁
  • 然后奇数线程获取到锁,执行
  • 这样交互执行到数值达到100就终止,break
  • 三个要素:
复制代码
  第一步:while循环判断当前数字是否小于等于100,
复制代码
  第二步:尝试获取到锁
复制代码
  第三步: 根据数字是奇数还是偶数判断,是打印数字,并累加,然后唤醒其他线程,还是当前线程直接wait方法,释放锁
java 复制代码
public class AlternatePrinting {
    private static final int MAX_NUMBER = 100;
    private static int currentNumber = 1;
    private static final Object lock = new Object();

    public static void main(String[] args) {
        Thread oddThread = new Thread(() -> {
            while (currentNumber <= MAX_NUMBER) {
                synchronized (lock) {
                    // // 让偶数线程打印,且未超过最大值
                    while (currentNumber <= MAX_NUMBER && currentNumber % 2 != 1) {
                        try {
                            //当前线程使当前线程进入等待状态,并且释放锁
                            lock.wait();
                        } catch (InterruptedException e) {
                            Thread.currentThread().interrupt();
                        }
                    }
                    if (currentNumber > MAX_NUMBER) break;
                    System.out.println("Odd: " + currentNumber++);
                    lock.notifyAll(); // 唤醒偶数线程
                }
            }
        });

        Thread evenThread = new Thread(() -> {
            while (currentNumber <= MAX_NUMBER) {
                synchronized (lock) {
                    // 让奇数线程打印,且未超过最大值
                    while (currentNumber <= MAX_NUMBER && currentNumber % 2 == 1) {
                        try {
                            lock.wait(); // 等待奇数线程打印后唤醒
                        } catch (InterruptedException e) {
                            Thread.currentThread().interrupt();
                        }
                    }
                    if (currentNumber > MAX_NUMBER) break;
                    System.out.println("Even: " + currentNumber++);
                    lock.notifyAll(); // 唤醒奇数线程
                }
            }
        });

        oddThread.start();
        evenThread.start();
    }
}
相关推荐
D***77651 小时前
适用于IntelliJ IDEA 2024.1.2部署Tomcat的完整方法,以及笔者踩的坑,避免高血压,保姆级教程
java·tomcat·intellij-idea
豆浆whisky1 小时前
Go并发模式选择指南:找到最适合你项目的并发方案|Go语言进阶(19)
开发语言·后端·golang
20岁30年经验的码农1 小时前
Spring Security 6.x 安全实践指南
java
草莓熊Lotso1 小时前
《算法闯关指南:动态规划算法--斐波拉契数列模型》--01.第N个泰波拉契数,02.三步问题
开发语言·c++·经验分享·笔记·其他·算法·动态规划
胖咕噜的稞达鸭2 小时前
自定义shell命令行解释器自制
java·开发语言
草莓熊Lotso2 小时前
Git 分支管理:从基础操作到协作流程(本地篇)
大数据·服务器·开发语言·c++·人工智能·git·sql
报错小能手2 小时前
C++异常处理 终极及总结
开发语言·c++
q***33375 小时前
oracle 12c查看执行过的sql及当前正在执行的sql
java·sql·oracle
tobebetter95277 小时前
How to manage python versions on windows
开发语言·windows·python
Y***h1878 小时前
第二章 Spring中的Bean
java·后端·spring