淘宝面试题(wait,notify解决线程间通信)

                      淘宝面试题(wait,notify解决线程间通信)
package com.qqjx.thread;

/**
 * 淘宝面试题:
 * 实现一个容器,提供两个方法,add,size
 * 写两个线程,线程1添加10个元素到容器中,线程2实现监控元素的个数,当个数到5个时,线程2给出提示并结束
 *
 * 给lists添加volatile之后,t2能够接到通知,但是,t2线程的死循环很浪费cpu,如果不用死循环,该怎么做呢?
 *
 * 这里使用wait和notify做到,wait会释放锁,而notify不会释放锁
 * 需要注意的是,运用这种方法,必须要保证t2先执行,也就是首先让t2监听才可以
 *
 * 阅读下面的程序,并分析输出结果
 * 可以读到输出结果并不是size=5时t2退出,而是t1结束时t2才接收到通知而退出
 * 想想这是为什么?
 *
 * notify之后,t1必须释放锁,t2退出后,也必须notify,通知t1继续执行
 * 整个通信过程比较繁琐
 * @author
 */

import java.util.ArrayList;
import java.util.List;
import java.util.concurrent.TimeUnit;

public class T04_NotifyFreeLock {
    
    

    //添加volatile,使t2能够得到通知
    volatile List lists = new ArrayList();

    public void add(Object o) {
    
    
        lists.add(o);
    }

    public int size() {
    
    
        return lists.size();
    }

    public static void main(String[] args) {
    
    
        T04_NotifyFreeLock c = new T04_NotifyFreeLock();

        final Object lock = new Object();

        new Thread(() -> {
    
    
            synchronized(lock) {
    
    
                System.out.println("t2启动");
                if(c.size() != 5) {
    
    
                    try {
    
    
                        //释放锁 线程阻塞
                        lock.wait();
                    } catch (InterruptedException e) {
    
    
                        e.printStackTrace();
                    }
                }
                System.out.println("t2 结束");
                //通知t1继续执行 唤醒t1线程
                lock.notify();
            }

        }, "t2").start();

        try {
    
    
            TimeUnit.SECONDS.sleep(1);
        } catch (InterruptedException e1) {
    
    
            e1.printStackTrace();
        }

        new Thread(() -> {
    
    
            System.out.println("t1启动");
            synchronized(lock) {
    
    
                for(int i=0; i<10; i++) {
    
    
                    c.add(new Object());
                    System.out.println("add " + i);

                    if(c.size() == 5) {
    
    
                        //唤醒t2线程
                        lock.notify();
                        //释放锁,让t2得以执行
                        try {
    
    
                            lock.wait();
                        } catch (InterruptedException e) {
    
    
                            e.printStackTrace();
                        }
                    }

                    try {
    
    
                        TimeUnit.SECONDS.sleep(1);
                    } catch (InterruptedException e) {
    
    
                        e.printStackTrace();
                    }
                }
            }
        }, "t1").start();


    }

}

t2启动
t1启动
add 0
add 1
add 2
add 3
add 4
t2 结束
add 5
add 6
add 7
add 8
add 9

猜你喜欢

转载自blog.csdn.net/m0_52936310/article/details/113172152
今日推荐