J.U.C源码阅读系列-TheadPoolExecutor

ThreadPoolExecutor类继承体系:

ThreadPoolExecutor -> AbstractExecutorService -> ExecutorService -> Executor

Executors提供工厂方法,用于创建不同类型的线程池
  • newFixedThreadPool()
  • newSingleThreadExecutor()
  • newCachedThreadPool()
  • newScheduledThreadPool()
    前两种线程池可能队列积压导致OOM,后两种可能创建非常多的线程(Integer.MAX_VALUE),甚至OOM

以下内容借鉴:
【链接】如何优雅的关闭Java线程池
http://url.cn/5DGL2xe


线程的状态

  • NEW
  • RUNNABLE(RUNNING)
  • BLOCKED
  • WAITING
  • TIMED_WAIT
  • TERMINATED

1. 新建(new) :新创建了一个线程对象。
2. 可运行(runnable) :线程对象创建后,其他线程(比如main线程)调用了该对象的start()方法。该状态的线程位于可运行线程池中,等待被线程调度选中,获取cpu 的使用权 。
3. 运行(running) :可运行状态( runnable) 的线程获得了cpu 时间片(timeslice) ,执行程序代码。
4. 阻塞(block) :阻塞状态是指线程因为某种原因放弃了cpu 使用权,也即让出了cpu timeslice,暂时停止运行。直到线程进入 可运行(runnable) 状态,才有机会再次获得cpu timeslice 转到 运行(running) 状态。阻塞的情况分三种: (一). 等待阻塞: 运行(running) 的线程执行o.wait()方法,JVM会把该线程放入等待队列(waitting queue)中。 (二). 同步阻塞: 运行(running) 的线程在获取对象的同步锁时,若该同步锁被别的线程占用,则JVM会把该线程放入锁池(lock pool)中。 (三). 其他阻塞: 运行(running) 的线程执行Thread.sleep(long ms)或t.join()方法,或者发出了I/O请求时,JVM会把该线程置为阻塞状态。当sleep()状态超时、join()等待线程终止或者超时、或者I/O处理完毕时,线程重新转入 可运行(runnable) 状态。
5. 死亡(dead) :线程run()、main() 方法执行结束,或者因异常退出了run()方法,则该线程结束生命周期。死亡的线程不可再次复生。


执行任务

  • execute()
  • submit()
    可简单理解为:前者是不需要返回值,后者需要返回值(Future.get()),虽然都是异步处理。

关闭线程池

  • shutdownNow()
            checkShutdownAccess();
            advanceRunState(STOP);
            interruptWorkers();
            tasks = drainQueue();

1. 拒绝接收新提交的任务
2. 同时立刻关闭线程池,池中的任务不再执行
3. 小结:当我们调用线程池的shutdownNow时,如果线程正在getTask方法中执⾏,则会通过for循环进入到if语句,于是getTask 返回null,从而线程退出。不管线程池⾥是否有未完成的任务。如果线程因为执行提交到线程池里的任务而处于阻塞状态,则会导致报错。(如果任务里没有捕获InterruptedException异常),否则线程会执行完当前任务,然后通过getTask方法返回为null来退出。

  • shutdown()
            checkShutdownAccess();
            advanceRunState(SHUTDOWN);
            interruptIdleWorkers();
            onShutdown(); // hook for ScheduledThreadPoolExecutor

1. 拒绝接收新提交的任务
2. 同时等待池中的任务执行完毕后关闭线程池
3. 小结:当我们调用线程池的shutdownNow时,如果线程正在getTask方法中执⾏,则会通过for循环进入到if语句,于是getTask 返回null,从而线程退出。不管线程池⾥是否有未完成的任务。如果线程因为执行提交到线程池里的任务而处于阻塞状态,则会导致报错。(如果任务里没有捕获InterruptedException异常),否则线程会执行完当前任务,然后通过getTask方法返回为null来退出。


总结

  • shutdownNow(),可能会引起报错,可能会使线程池关闭不了,但不一定。(isTerminated=false)
  • shutdown(),要保证任务里不会有永久阻塞等待的逻辑,否则线程池也关闭不了。(isTerminated=false)

TestCase

        // 1. 线程正在执行时,shutdownNow不会报错
        final ExecutorService pool1 = Executors.newSingleThreadExecutor();
        pool1.submit(() -> {
            long startTime = System.currentTimeMillis();
            System.out.println("enter the thread pool1");
            long counter = 0L;
            while (counter < Long.MAX_VALUE >> (Integer.SIZE - 4)) {
                Object o = new Object();
                o = null;
                counter++;
            }
            System.out.println("weak up from sleeping pool1");
            System.out.println("leave the thread pool1");
            long finishTime = System.currentTimeMillis();
            System.out.println("cost time " + (finishTime - startTime));
        });
        System.out.println("going to shutdown the pool1");
        pool1.shutdownNow();
        pool1.awaitTermination(1, TimeUnit.MINUTES);
        System.out.println("exit the pool1");
        boolean flag = pool1.isTerminated();
        System.out.println("pool1->" + flag);

        System.out.println("=========================");

        // 2. 线程阻塞时,shutdownNow将会抛出InterruptedException
        final ExecutorService pool2 = Executors.newSingleThreadExecutor();
        pool2.submit(() -> {
            System.out.println("enter the thread pool2");
            try {
                // 为什么两种不同进入阻塞状态的实现,最终结果不一致
                //Thread.currentThread().wait(100_000);
                Thread.sleep(100_000);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            System.out.println("weak up from sleeping pool2");
            System.out.println("leave the thread pool2");
        });
        System.out.println("going to shutdown the pool2");
        pool2.shutdownNow();
        pool2.awaitTermination(1, TimeUnit.MINUTES);
        System.out.println("exit the pool2");
        flag = pool2.isTerminated();
        System.out.println("pool2->" + flag);

        System.out.println("=========================");

        // 3. 线程永久阻塞,shutdown将无法关闭线程池
        final ExecutorService pool3 = Executors.newSingleThreadExecutor();
        pool3.submit(() -> {
            System.out.println("enter the thread pool3");
//            for (long l = 0L; l < Long.MAX_VALUE; l++) {
//                Object o = new Object();
//                o = null;
//            }
//            try {
//                Thread.sleep(1000_000_000); // 模拟永久阻塞
//                //Thread.currentThread().wait();
//            } catch (InterruptedException e) {
//                e.printStackTrace();
//            }
            try {
                ServerSocket socket = new ServerSocket(8080);
                socket.accept();
            } catch (IOException e) {
                e.printStackTrace();
            }
            System.out.println("weak up from sleeping pool3");
            System.out.println("leave the thread pool3");
        });
        System.out.println("going to shutdown the pool3");
        pool3.shutdownNow();
        pool3.awaitTermination(1, TimeUnit.MINUTES);
        System.out.println("exit the pool3");
        flag = pool3.isTerminated();
        System.out.println("pool3->" + flag);
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 204,732评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 87,496评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 151,264评论 0 338
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,807评论 1 277
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,806评论 5 368
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,675评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,029评论 3 399
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,683评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 41,704评论 1 299
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,666评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,773评论 1 332
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,413评论 4 321
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,016评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,978评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,204评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,083评论 2 350
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,503评论 2 343

推荐阅读更多精彩内容