• java 多线程&使用wait/notify机制实现list.size()等于5时 的线程销毁——65


    下面通过一个示例来演示如何使用 wait() 与notify() 来实现前面 list.size() 值等于 5 时的线程销毁

    MyList.java代码

    1. package chapter3.test3_1.test3_1_6;
    2. import java.util.ArrayList;
    3. import java.util.List;
    4. public class MyList {
    5. private static List list = new ArrayList();
    6. public static void add() {
    7. list.add("anyString");
    8. }
    9. public static int size() {
    10. return list.size();
    11. }
    12. }

    ThreadA.java代码

    1. package chapter3.test3_1.test3_1_6;
    2. public class ThreadA extends Thread {
    3. private Object lock;
    4. public ThreadA(Object lock) {
    5. super();
    6. this.lock = lock;
    7. }
    8. @Override
    9. public void run() {
    10. try {
    11. synchronized (lock) {
    12. if (MyList.size() != 5) {
    13. System.out.println("wait begin " + System.currentTimeMillis());
    14. lock.wait();
    15. System.out.println("wait end " + System.currentTimeMillis());
    16. }
    17. }
    18. } catch (InterruptedException e) {
    19. e.printStackTrace();
    20. }
    21. }
    22. }

    ThreadB.java代码

    1. package chapter3.test3_1.test3_1_6;
    2. public class ThreadB extends Thread {
    3. private Object lock;
    4. public ThreadB(Object lock) {
    5. super();
    6. this.lock = lock;
    7. }
    8. @Override
    9. public void run() {
    10. try {
    11. synchronized (lock) {
    12. for (int i = 0; i < 10; i++) {
    13. MyList.add();
    14. if (MyList.size() == 5) {
    15. lock.notify();
    16. System.out.println("已发出通知!");
    17. }
    18. System.out.println("添加了" + (i + 1) + "个元素!");
    19. Thread.sleep(1000);
    20. }
    21. }
    22. } catch (InterruptedException e) {
    23. e.printStackTrace();
    24. }
    25. }
    26. }

    Run.java代码

    1. package chapter3.test3_1.test3_1_6;
    2. public class Run {
    3. public static void main(String[] args) {
    4. try {
    5. Object lock = new Object();
    6. ThreadA a = new ThreadA(lock);
    7. a.start();
    8. Thread.sleep(50);
    9. ThreadB b = new ThreadB(lock);
    10. b.start();
    11. } catch (InterruptedException e) {
    12. e.printStackTrace();
    13. }
    14. }
    15. }

    程序运行结果

    日志信息 wait end 在最后输出,这说明notify()方法 执行后并不立即释放锁 
            关键字synchronized 可以将任何一个Object对象作为锁来看待,而Java为每个Object都实现了wait()和notify()方法,它们必须用在被synchronized同步的Object的临界区内。通过调用wait()方法可以使处于临界区内的线程进入等待状态,同时释放被同步对象的锁,而notify操 作可以唤醒一个因调用了wait操作而处于wait状态中的线程,使其进入就绪状态,被重新唤醒的线程会试图重新获得临界区的控制权,也就是锁,并继续执行临界区wait之后的代码。如果发出notify操作时没有处于wait状态中的线程,那么该命令会被忽略。
           wait()方法可以使调用该方法的线程释放锁,然后从运行状态转换成wait状态,等待被唤醒。

            notify()方法按照执行wait()方法的顺序唤醒等待同一锁的“一个”线程,使其进入可运行状态,即notify()方法仅通知“一个”线程。

            notifyAll()方法执行后,会按照执行wait()方法相反的顺序依次唤醒全部的线程。

     

  • 相关阅读:
    高速DSP系统设计参考指南(一)高速DSP设计面临的挑战
    QT多线程的四种实现方式之一:QThread类的run
    简单的购物车
    10min快速回顾C++语法(三)
    可变字符串
    比 N 小的最大质数
    centos7安装chrome/Firefox浏览器
    GDB Debugging Notes
    Qt listWidget 详细分析
    基于qt软件的网上聊天室软件
  • 原文地址:https://blog.csdn.net/zp357252539/article/details/125460060