java多线程之wait和notify

多线程中的通信是非常重要的概念,线程直接实现通信就可以并发完成很多复杂工作。

java在Object类中就设计了wait()和notify()两个方法,以解决这个问题。

1.释义:

wait()方法将当前线程暂停,置于“预执行队列”中,而notify()则用于通知一个在wait等待中的线程,可以继续执行了

2.wait()和notify()的使用条件:

wait()和notify()必须置于同步方法和同步代码块中使用,即在调用前,线程必须获得该对象的对象级别锁。否则会抛出异常

3.线程执行到wait()时,直接释放锁,处于等待状态;

复制代码
public class WaitService {
    public void testMethod(Object lock) {
        try {
            synchronized (lock) {
                System.out.println("wait begin");
                lock.wait();
                System.out.println("wait end");
            }
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
    }
}

public class WaitThread1 extends Thread {
    private Object lock;

    public WaitThread1(Object lock) {
        this.lock = lock;
    }

    @Override
    public void run() {
        WaitService service = new WaitService();
        service.testMethod(lock);
    }
}

public class WaitThread2 extends Thread {
    private Object lock;

    public WaitThread2(Object lock) {
        this.lock = lock;
    }

    @Override
    public void run() {
        WaitService service = new WaitService();
        service.testMethod(lock);
    }
}

public class WaitTest {
    public static void main(String[] args) {
        Object lock=new Object();
        WaitThread1 t1=new WaitThread1(lock);
        WaitThread2 t2 = new WaitThread2(lock);
        t1.start();
        t2.start();
    }
}
复制代码

执行结果:

wait begin
wait begin

4.线程执行notify时,当前线程并不马上释放锁,wait状态的线程也不会马上获得锁,要执行notify方法的程序执行完,退出syncronized代码块后,才释放锁。

复制代码
public class NotifyService {
    public void testMethod(Object lock){
        try {
            synchronized (lock){
                System.out.println("wait  begin");
                lock.wait();
                System.out.println("wait  end");
            }
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
    }

    public void notifyMethod(Object lock){
        try {
            synchronized (lock) {
                System.out.println("begin nofity ");
                Thread.sleep(1000);
                lock.notify();
                System.out.println("end nofity ");
            }
        } catch (Exception e) {
            e.printStackTrace();
        }
    }
}

public class NotifyThreadA extends Thread{
    private Object lock;

    public NotifyThreadA(Object lock) {
        this.lock = lock;
    }

    @Override
    public void run() {
        NotifyService service=new NotifyService();
        service.testMethod(lock);
    }
}

public class NotifyThreadB extends Thread {
    private Object lock;

    public NotifyThreadB(Object lock) {
        this.lock = lock;
    }

    @Override
    public void run() {
        NotifyService service=new NotifyService();
        service.notifyMethod(lock);
    }
}

public class NotifyTest {
    public static void main(String[] args) {
        Object lock=new Object();
        NotifyThreadA a = new NotifyThreadA(lock);
        a.start();
        NotifyThreadB b = new NotifyThreadB(lock);
        b.start();
    }
}
复制代码

输出:

wait  begin
begin nofity 
end nofity 
wait  end

 

posted @   Mars.wang  阅读(440)  评论(0编辑  收藏  举报
编辑推荐:
· AI与.NET技术实操系列:基于图像分类模型对图像进行分类
· go语言实现终端里的倒计时
· 如何编写易于单元测试的代码
· 10年+ .NET Coder 心语,封装的思维:从隐藏、稳定开始理解其本质意义
· .NET Core 中如何实现缓存的预热?
阅读排行:
· 25岁的心里话
· 闲置电脑爆改个人服务器(超详细) #公网映射 #Vmware虚拟网络编辑器
· 零经验选手,Compose 一天开发一款小游戏!
· 通过 API 将Deepseek响应流式内容输出到前端
· AI Agent开发,如何调用三方的API Function,是通过提示词来发起调用的吗
点击右上角即可分享
微信分享提示