淘宝面试题(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才接收到通知而退出
 * 想想这是为什么?
 * @author
 */

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

public class T03_NotifyHoldingLock {
    
    

    //添加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) {
    
    
        T03_NotifyHoldingLock c = new T03_NotifyHoldingLock();

        final Object lock = new Object();

        new Thread(() -> {
    
    
            synchronized(lock) {
    
    
                System.out.println("t2启动");
                if(c.size() != 5) {
    
    
                    try {
    
    
                        //线程阻塞 但是得拿到lock这把锁才可以继续执行程序
                        lock.wait();
                    } catch (InterruptedException e) {
    
    
                        e.printStackTrace();
                    }
                }
                System.out.println("t2 结束");
            }

        }, "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) {
    
    
                        //将另外阻塞的线程叫醒 但是notify()并不释放锁
                        lock.notify();
                    }

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


    }

}

t2启动
t1启动
add 0
add 1
add 2
add 3
add 4
add 5
add 6
add 7
add 8
add 9
故这种方式得不到想要的结果

猜你喜欢

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