Java使用wait和nofify方法控制线程的执行状态

最近项目中遇到了一个特殊的问题,需要使用异步的方式完成某个小任务,还要求异步任务完成之后返回的结果集,由发起任务的线程继续去处理。这样的状况用回调的方式去处理,需要切换线程,可能不太合适。笔者想到这个问题应该比较类似于操作系统里面的生产者消费者问题,就决定借鉴一下生产者消费者问题的解决思路。线程A和线程B共享同一块内存区,线程A缺少执行锁需要的资源时,向线程B发送消息,让线程B去获取资源,发完消息之后,线程A进入阻塞状态,线程B得到消息之后,从阻塞态变为执行状态,执行完成之后,将结果集放入两个线程共享的内存区,然后向线程A发送消息,让其继续执行,若此时线程B没有其他任务需要执行,便进入阻塞状态。

一、Object.wait()
线程执行执行实例对象的wait()方法之前,必须获取到该对象的锁,执行到实例对象的wait()方法之后,会释放该对象的锁,然后进入阻塞状态。线程A就是扮演的这个角色。示例代码如下:

        synchronized (obj) {
            try {
                obj.wait();

            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }

*** 二、obj.notify()***
线程执行实例对象的notify()方法之前,也需要获取到该对象的锁,执行完该对象的notify()方法,并且释放锁之后,会唤醒因为该对象而阻塞的线程之一,具体释放哪个线程就要看线程调度怎么安排了。线程B就是扮演的这个角色。示例代码如下:

        synchronized (obj) {
            obj.notify();
        }

三、趟坑第一步
考虑到我这个项目很可能经常会遇到这种控制线程执行状态的问题,所以我决定创建一个能管理所有线程的类Manager,让它维持一个标志位对象的Map,这样以后遇到类似的问题,就不用再麻烦了。我们创建一个MonitorObject类,用它来产生标志位对象,直接用Object类也完全可以。因为我们不需要什么属性,只要wait()和notify()方法可以用就行。

public class Manager {

    public static Map<String, Object> map = new HashMap<String, Object>();

    public static void create(String key) {
        if (map.get(key) == null) {
            Object monitorObject = new Object();
            map.put(key, monitorObject);
        }

    }

    public static void doWait(String key) {
        create(key);
        Object obj = map.get(key);
        if (obj == null) return;
        synchronized (obj) {
            try {
                obj.wait();

            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }

  }

    public static void doNotify(String key) {
        Object obj = map.get(key);
        if (obj == null) return;
        synchronized (obj) {
            obj.notify();
        }
    }

}

四、线程调用
线程A在调起线程B执行异步任务之后,使用约定好的一个作为标志的key值,调用Manager的doWait()方法,线程B执行完任务之后,调用Manager的doNotify()方法。示例如下:

class ThreadA extends Thread {
     public void run() {
            /*
                唤起线程B执行异步任务
            */
            //线程A执行完doWait()方法后进入阻塞状态
            Manager.doWait("missonFlag");
           /*
                线程A唤醒之后的操作
            */
        }
    }

class ThreadB extends Thread {
     public void run() {
            /*
                执行子任务
            */
            Manager.doNotify("missonFlag");
        }
    }


五、发现问题后进行改造

上面是趟坑的第一步,写这段程序的时候,我刚毕业入职不到两个月,当时还感觉自己棒棒哒,这逻辑看上去完美无瑕。确实,最开始用的时候,确实还算正常,直到有几次发现线程A有时候唤醒不了,一开始还以为是线程B异步任务执行时遇到了网络问题,最后一步步的调试才发现,原来有的时候,doNotify()方法执行在了doWait()方法之前。这样的情况下,线程B执行doNotify()方法的时候,由于线程A还没有被阻塞,所以不会有任何作用。因为notify()方法的作用是在因为该实例对象而阻塞的线程中,唤醒一个线程去执行,线程A并不符合条件。但是线程A还是会顺序执行doWait()方法,这样线程A就被阻塞了,而且也不会受到唤醒它的消息,它就一直死等了,可怜的线程A!于是乎,我把Manager类进行了改造。
现在发现只用Object是不行了,必须添加一个属性才能行,需要添加一个计数器,用来记录等待的状态,执行 obj.wait(),则计数器加1,执行obj.notify()方法,计数器减1。其中还出现过一个小插曲,就是doNotify()方法执行在了doWait()方法之前的时候,MonitorObject的实例也没有被创建,所以把create()方法也单独拿了出来,在线程A唤起异步任务之前调用。

class MonitorObject {
    public int count = 0;
}

public class Manager {
public static Map<String, MonitorObject> map = new HashMap<String, MonitorObject>();

    public static void create(String key) {
        if (map.get(key) == null) {
            MonitorObject monitorObject = new MonitorObject();
            map.put(key, monitorObject);
        }
    }

    public static void doWait(String key) {

        MonitorObject obj = map.get(key);
        if (obj == null) return;
        synchronized (obj) {
            try {
                if (obj.count < 0){
                    obj.count++;
                    return;
                }
                obj.count++;
                obj.wait();
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }
    }

    public static void doNotify(String key) {

        MonitorObject obj = map.get(key);
        if (obj == null) return;
        synchronized (obj) {
            obj.count--;
            obj.notify();
        }
    }
}

六、改造之后使用Manager
改造之后,线程A在使用Manager之前,必须先创建MonitorObject对象,这一步是为了保证不管doWait()和doNotify()哪个先执行,计数器的作用能正常发挥。改造之后的Manager仍然不能控制doWait()和doNotify()哪个先执行,但是如果doNotify()先于doWait()执行,在执行doWait()操作时,线程不会阻塞,只会把MonitorObject的计数器做自增操作。因为doNotify()先执行,说明异步任务已经完成,线程A已经无需等待了。
示例代码如下:

class ThreadA extends Thread {
     public void run() {
            Manager.create("missonFlag");
            /*
                唤起线程B执行异步任务
            */
            //线程A执行完doWait()方法后进入阻塞状态
            Manager.doWait("missonFlag");
           /*
                线程A唤醒之后的操作
            */
        }
    }

class ThreadB extends Thread {
     public void run() {
            /*
                执行子任务
            */
            Manager.doNotify("missonFlag");
        }
    }
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,711评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,079评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,194评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,089评论 1 286
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,197评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,306评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,338评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,119评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,541评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,846评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,014评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,694评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,322评论 3 318
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,026评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,257评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,863评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,895评论 2 351

推荐阅读更多精彩内容

  • Java多线程学习 [-] 一扩展javalangThread类 二实现javalangRunnable接口 三T...
    影驰阅读 2,954评论 1 18
  • 该文章转自:http://blog.csdn.net/evankaka/article/details/44153...
    加来依蓝阅读 7,340评论 3 87
  • 本文主要讲了java中多线程的使用方法、线程同步、线程数据传递、线程状态及相应的一些线程函数用法、概述等。 首先讲...
    李欣阳阅读 2,448评论 1 15
  • 我的商女梦 先说明一下,最初受启...
    拙兰阅读 359评论 1 4
  • 今天最大的收获就是这句话,授人以鱼不如授人以渔。早上看书发现了自己身上的毛病,从今天上午到下午的工作,真的验证了我...
    左耳不闻阅读 247评论 0 0