您的位置:首页 > 科技 > IT业 > 线程通信【详解】

线程通信【详解】

2024/11/16 5:29:14 来源:https://blog.csdn.net/weixin_61898502/article/details/141401846  浏览:    关键词:线程通信【详解】

线程通信

线程通信是指线程间可以交互,指定信号,让线程执行或者等待

通过Object类中的方法完成通信

  • wait()

  • notify()

1、 同步方法

需求: 两个输出的方法,保证正常输出不被打断且达到一人一次输出的效果

public class Printer {
​// 定义一个信号量// 1代表print1执行 2代表print2执行private int flag = 1;
​/*** 线程通信的要求* 1) 要保证线程安全* 2) 线程等待方法是wait*    线程唤醒方法是notify* 3) 必须使用锁对象调用 通信的方法* ---------* 为什么,wait和notify这些线程通信的方法要设计在Object类?* 答:*/public synchronized void print1() throws InterruptedException {if (flag != 1){ // 信号不是1,说明不该print1执行,那就等待this.wait();}System.out.print(1 + " ");System.out.print(2 + " ");System.out.print(3 + " ");System.out.print(4 + " ");System.out.println( );
​// 改变信号量flag = 2;// 通知处于等待状态的线程启动this.notify();}public synchronized void print2() throws InterruptedException {if (flag != 2) { // 信号不是2,说明不该print2执行,那就等待this.wait();}System.out.print("A ");System.out.print("B ");System.out.print("C ");System.out.print("D ");System.out.println( );
​flag = 1;// 通知处于等待状态的线程启动this.notify();}
}
public class TestPrinter {
​public static void main(String[] args) {Printer p = new Printer( );// 开启一个线程new Thread(){@Overridepublic void run() {while(true){try {p.print1();} catch (InterruptedException e) {throw new RuntimeException(e);}}}}.start();
​// 又开启一个线程new Thread(){@Overridepublic void run() {while(true){try {p.print2();} catch (InterruptedException e) {throw new RuntimeException(e);}}}}.start();}
​
}

2、 同步代码块

再使用同步代码块演示一遍,再次确定一个结论

  • 锁对象是谁,就使用哪个对象来调用wait和notify

public class Printer2 {
​// 定义一个信号量// 1代表print1执行 2代表print2执行private int flag = 1;
​/*** 线程通信的要求* 1) 要保证线程安全* 2) 线程等待方法是wait*    线程唤醒方法是notify* 3) 必须使用锁对象调用 通信的方法* ---------* 为什么,wait和notify这些线程通信的方法要设计在Object类?* 答:*/public void print1() throws InterruptedException {synchronized (Object.class) {if (flag != 1) { // 信号不是1,说明不该print1执行,那就等待Object.class.wait( );}System.out.print(1 + " ");System.out.print(2 + " ");System.out.print(3 + " ");System.out.print(4 + " ");System.out.println( );
​// 改变信号量flag = 2;// 通知处于等待状态的线程启动Object.class.notify( );}}public void print2() throws InterruptedException {synchronized (Object.class) {if (flag != 2) { // 信号不是2,说明不该print2执行,那就等待Object.class.wait( );}System.out.print("A ");System.out.print("B ");System.out.print("C ");System.out.print("D ");System.out.println( );
​flag = 1;// 通知处于等待状态的线程启动Object.class.notify( );}}
}

补充: 目前这个代码可以保证两个线程通信,如果>= 3个线程,就不一定能按照预想顺序完成

原因是,线程过多,但是notify方法只能随机唤醒一个处于等待状态的线程

解决方案: 使用notifyAll

3、 wait和sleep

  • wait

    • 是Object类中的方法

    • wait会让线程等待

    • wait方法必须在同步方法中使用

    • wait方法方法线程等待时,会让出资源,别的线程可以执行

  • sleep

    • 是Thread类中的方法

    • sleep会让线程等待

    • 方法同步或者不同步都可以使用

      • 如果线程不安全,使用了sleep,会让出资源,别的线程执行

      • 如果线程安全,使用了sleep,不会释放资源,别的线程不会执行,会阻塞 --> 抱着锁睡

版权声明:

本网仅为发布的内容提供存储空间,不对发表、转载的内容提供任何形式的保证。凡本网注明“来源:XXX网络”的作品,均转载自其它媒体,著作权归作者所有,商业转载请联系作者获得授权,非商业转载请注明出处。

我们尊重并感谢每一位作者,均已注明文章来源和作者。如因作品内容、版权或其它问题,请及时与我们联系,联系邮箱:809451989@qq.com,投稿邮箱:809451989@qq.com