• 深入理解Java多线程之线程间的通信方式(上)


    Java线程间的通信方式

    这篇文章来总结下我对JAVA多线程中线程之间的通信方式的理解,主要以代码结合文字的方式来讨论线程间的通信,故摘抄了书中的一些示例代码。

    Java线程间的通信方式

    1、同步

    这里讲的同步是指多个线程通过synchronized关键字这种方式来实现线程间的通信。

    1. public class MyObject {
    2. synchronized public void methodA() {
    3. //do something....
    4. }
    5. synchronized public void methodB() {
    6. //do some other thing
    7. }
    8. }
    9. 复制代码
    1. public class ThreadA extends Thread {
    2. private MyObject object;
    3. //省略构造方法
    4. @Override
    5. public void run() {
    6. super.run();
    7. object.methodA();
    8. }
    9. }
    10. 复制代码
    1. public class ThreadB extends Thread {
    2. private MyObject object;
    3. //省略构造方法
    4. @Override
    5. public void run() {
    6. super.run();
    7. object.methodB();
    8. }
    9. }
    10. 复制代码
    1. public class Run {
    2. public static void main(String[] args) {
    3. MyObject object = new MyObject();
    4. //线程A与线程B 持有的是同一个对象:object
    5. ThreadA a = new ThreadA(object);
    6. ThreadB b = new ThreadB(object);
    7. a.start();
    8. b.start();
    9. }
    10. }
    11. 复制代码

    由于线程A和线程B持有同一个MyObject类的对象object,尽管这两个线程需要调用不同的方法,但是它们是同步执行的,比如:线程B需要等待线程A执行完了methodA()方法之后,它才能执行methodB()方法。这样,线程A和线程B就实现了通信。这种方式,本质上就是“共享内存”式的通信。多个线程需要访问同一个共享变量,谁拿到了锁(获得了访问权限),谁就可以执行。

    2、while轮询的方式

    代码如下:

    1. import java.util.ArrayList;
    2. import java.util.List;
    3. public class MyList {
    4. private List list = new ArrayList();
    5. public void add() {
    6. list.add("elements");
    7. }
    8. public int size() {
    9. return list.size();
    10. }
    11. }
    12. 复制代码
    1. import mylist.MyList;
    2. public class ThreadA extends Thread {
    3. private MyList list;
    4. public ThreadA(MyList list) {
    5. super();
    6. this.list = list;
    7. }
    8. @Override
    9. public void run() {
    10. try {
    11. for (int i = 0; i < 10; i++) {
    12. list.add();
    13. System.out.println("添加了" + (i + 1) + "个元素");
    14. Thread.sleep(1000);
    15. }
    16. } catch (InterruptedException e) {
    17. e.printStackTrace();
    18. }
    19. }
    20. }
    21. 复制代码
    1. import mylist.MyList;
    2. public class ThreadB extends Thread {
    3. private MyList list;
    4. public ThreadB(MyList list) {
    5. super();
    6. this.list = list;
    7. }
    8. @Override
    9. public void run() {
    10. try {
    11. while (true) {
    12. if (list.size() == 5) {
    13. System.out.println("==5, 线程b准备退出了");
    14. throw new InterruptedException();
    15. }
    16. }
    17. } catch (InterruptedException e) {
    18. e.printStackTrace();
    19. }
    20. }
    21. }
    22. 复制代码
    1. import mylist.MyList;
    2. import extthread.ThreadA;
    3. import extthread.ThreadB;
    4. public class Test {
    5. public static void main(String[] args) {
    6. MyList service = new MyList();
    7. ThreadA a = new ThreadA(service);
    8. a.setName("A");
    9. a.start();
    10. ThreadB b = new ThreadB(service);
    11. b.setName("B");
    12. b.start();
    13. }
    14. }
    15. 复制代码

    在这种方式下,线程A不断地改变条件,线程ThreadB不停地通过while语句检测这个条件(list.size()==5)是否成立 ,从而实现了线程间的通信。但是这种方式会浪费CPU资源。之所以说它浪费资源,是因为JVM调度器将CPU交给线程B执行时,它没做啥“有用”的工作,只是在不断地测试 某个条件是否成立。就类似于现实生活中,某个人一直看着手机屏幕是否有电话来了,而不是: 在干别的事情,当有电话来时,响铃通知TA电话来了。关于线程的轮询的影响,可参考:JAVA多线程之当一个线程在执行死循环时会影响另外一个线程吗?

    这种方式还存在另外一个问题:

    轮询的条件的可见性问题,关于内存可见性问题,可参考:JAVA多线程之volatile 与 synchronized 的比较中的第一点“一,volatile关键字的可见性”

    线程都是先把变量读取到本地线程栈空间,然后再去再去修改的本地变量。因此,如果线程B每次都在取本地的 条件变量,那么尽管另外一个线程已经改变了轮询的条件,它也察觉不到,这样也会造成死循环

  • 相关阅读:
    数据结构-字符串详解
    Nginx__高级进阶篇
    机器学习(三十六):随机梯度下降(SGD)算法
    【C++】string 之 substr、insert、erase函数的学习
    dom-to-image库是如何将html转换成图片的
    spring-boot-maven-plugin插件详解
    密码学消息鉴别
    设计模式之命令模式(Command)的C++实现
    图的初识·存储结构
    Spring中PointcutAdvisor和IntroductionAdvisor梳理
  • 原文地址:https://blog.csdn.net/BASK2312/article/details/128076001