线程池 --------常见的四中线程池

由于线程的频繁调度,而影响性能,通过线程池来维护,减少线程的频繁的创建和销毁。

在Executors统一管理,看一下常见的四中线程池:

1.newFixedThreadPool:创建定长的线程池,超出定长在线程队列中等待。

```

public static ExecutorService newFixedThreadPool(int nThreads) {

        return new ThreadPoolExecutor(nThreads, nThreads,

                                      0L, TimeUnit.MILLISECONDS,

                                      new LinkedBlockingQueue<Runnable>());

    }

```

2.newCachedThreadPool:线程数无限大,当线程队列中有空闲线程时会复用,否则重新创建线程,同时线程60s没有用时,从线程队列中移除。

```

public static ExecutorService newCachedThreadPool() {

        return new ThreadPoolExecutor(0, Integer.MAX_VALUE,

                                      60L, TimeUnit.SECONDS,

                                      new SynchronousQueue<Runnable>());

    }

```

3.newScheduledThreadPool:次线程池是在规定的时间进行调度。

```

public static ScheduledExecutorService newScheduledThreadPool(int corePoolSize) {

        return new ScheduledThreadPoolExecutor(corePoolSize);

    }

```

4.newSingleThreadExecutor:仅有一个线程在调度,且有顺序的进行出队和入队。

```

public static ExecutorService newSingleThreadExecutor() {

        return new FinalizableDelegatedExecutorService

            (new ThreadPoolExecutor(1, 1,

                                    0L, TimeUnit.MILLISECONDS,

                                    new LinkedBlockingQueue<Runnable>()));

    }

```

但是他们都调用了ThreadPoolExecutor:看一下核心构造器里的几个参数

corePoolSize: 要保留在池中的线程数量。

maximumPoolSize:线程池中允许的最大线程数。

keepAliveTime:空闲线程等待的最大时间。

unit:keepAliveTime的时间单位。

workQueue:可执行的任务队列。

threadFactory:执行创建一个新的线程时使用。

handler:当达到了线程边界和队列容量,执行被阻塞时使用的处理程序。

```

public ThreadPoolExecutor(int corePoolSize,

                              int maximumPoolSize,

                              long keepAliveTime,

                              TimeUnit unit,

                              BlockingQueue<Runnable> workQueue,

                              ThreadFactory threadFactory,

                              RejectedExecutionHandler handler) {

        if (corePoolSize < 0 ||

            maximumPoolSize <= 0 ||

            maximumPoolSize < corePoolSize ||

            keepAliveTime < 0)

            throw new IllegalArgumentException();

        if (workQueue == null || threadFactory == null || handler == null)

            throw new NullPointerException();

        this.corePoolSize = corePoolSize;

        this.maximumPoolSize = maximumPoolSize;

        this.workQueue = workQueue;

        this.keepAliveTime = unit.toNanos(keepAliveTime);

        this.threadFactory = threadFactory;

        this.handler = handler;

    }

```

线程池中常用几个方法源码:

看一下runState控制得几种状态:

```

RUNNING:  接受新任务并能够对添加的任务处理。

SHUTDOWN: 不接受新任务,但能处理已在队列中的任务。

STOP:    不接受新任务,不处理排队的任务,并同时中断正在进行的任务。

TIDYING:  所有任务全部中断,同时将workerCount 置0,并将所有线程切换到TIDYING状态,会执行钩子函数terminated()。

TERMINATED: 表示线程池彻底终止。执行完terminated()之后,就会由 TIDYING -> TERMINATED。

```

线程池的几种转态转换:

![在这里插入图片描述](https://img-blog.csdnimg.cn/20181219133330333.png?x-oss-process=image/watermark,type_ZmFuZ3poZW5naGVpdGk,shadow_10,text_aHR0cHM6Ly9ibG9nLmNzZG4ubmV0L3FxXzM0NDQ3MzI4,size_16,color_FFFFFF,t_70)

```

public void execute(Runnable command) {

        if (command == null)

            throw new NullPointerException();

        int c = ctl.get();//保证了数据的原子性,不被其他线程干扰

        if (workerCountOf(c) < corePoolSize) {//运行的线程数小于corePoolSize的数量时

            if (addWorker(command, true))//添加一个新的到任务中

                return;

            c = ctl.get();

        }

        if (isRunning(c) && workQueue.offer(command)) {//插入任务成功且是Running状态下

            int recheck = ctl.get();

            if (! isRunning(recheck) && remove(command))//移除任务成功且不是Running状态下

                reject(command);

            else if (workerCountOf(recheck) == 0)

                addWorker(null, false);

        }

        //如果添加到任务中失败,抛出异常。

        else if (!addWorker(command, false))

            reject(command);

    }

```

这里addWorker(Runnable firstTask, boolean core)中的两个参数:

firstTask:新线程首要运行。

core:用一个布尔值判断当前线程数是否小于corePoolSize或maximumPoolSize。在源码中

```

wc >= (core ? corePoolSize : maximumPoolSize))

```

shutdown()方法:对以前的任务切换到SHUTDOWN状态,不接受新的任务。

```

public void shutdown() {

        final ReentrantLock mainLock = this.mainLock;//重入锁

        mainLock.lock();

        try {

            checkShutdownAccess();//判断调用者是否有权限shutdown线程池

            advanceRunState(SHUTDOWN);//线程池状态切换为SHUTDOWN

            interruptIdleWorkers();//中断可能正在等待任务的线程

            onShutdown(); // SHUTDOWN转态下的进一步清理。但是使用了ScheduledThreadPoolExecutor取消延迟的任务。

        } finally {

            mainLock.unlock();

        }

        tryTerminate();//试着终止线程池

    }

```

更多文章:https://blog.csdn.net/qq_34447328

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

推荐阅读更多精彩内容