java_多线程——线程池、submit和execute区别

一、线程池的概念

        顾名思义就是事先准备一个池子(线程池),初始化一些线程,当使用可以快速调用,不用再初始化线程,使用完成后,不再销毁该线程,归还到线程池,方便后面复用。节省创建和销毁线程资源,提高线程使用效率。

二、线程池的好处

        1)节省资源

        2)提高线程执行效率

        3)提高了线程管理效率,更方便管理调度线程

三、线程池原理

3.1 ThreadPoolExecutor 七大核心参数

        1)int corePoolSize

        核心线程池大小 (核心线程最大数量)

        核心线程:线程池中有两类线程:核心线程和非核心线程。核心线程默认情况下会一直存在于线程中,即使这个核心线程什么都不干,而非核心线程如果长时间的闲置,就会被销毁。

        2)int maximumPoolSize

         最大线程池大小 (线程总数量最大值),该值等于核心线程数+非核心线程数

        3)long keepAliveTime

        线程池中超过corePoolSize数目的空闲线程最大存活时间(非核心线程的闲置超时时间)

        4)TimeUnit unit

        时间单位(keepAliveTime的时间单位)

        TimeUnit是一个枚举类型:MILLSECONDS:1毫秒、SECONDS:1秒、MINUTES:1分。。。

        5)BlockingQueue<Runnable> workQueue

        线程等待队列(阻塞队列)

        常见的几个阻塞队列:

        1、LinkedBlockingQueue

        链式阻塞队列,默认大小:Interger.MAX_VALUE,可以指定大小。

        2、ArrayBlockingQueue

        数组阻塞队列,需要指定大小

        3、SynchronousQueue

        同步队列,内部容量为0,每个put操作都必须等待一个take操作。反之亦然

        4、DelayQueue

        延迟队列,队列中的元素之后当其指定的延迟时间到了,才能从队列中获取到改元素。

        5、PriorityBlockingQueue

        优先阻塞队列:无界,默认采用元素自然顺序升序排列。

        6)ThreadFactory threadFactory

        线程工厂:创建线程的工厂,用于批量创建线程,如果不指定,会新建一个默认的线程工厂。

        7)RejectedExecutionHander handler

        拒绝处理策略,当无法创建新线程处理任务并且阻塞队列已满时就会采用拒绝处理策略。

        jdk默认四种策略:

        ThreadPoolExecutor.AbortPolicy:默认拒绝处理策略,丢弃任务并抛出异常

        ThreadPollExecutor.DiscardPolicy:丢弃新来的任务,但不抛出异常

        ThreadPollExecutor.DiscardOldestPolicy:丢弃阻塞队列头部(最旧)的任务,然后重新尝试执行程序,(如果再次失败,重复此过程)

        ThreadPollExecutor.CallerRunPolicy:由调用的线程去处理改任务。只适用于并发小的情况。

3.2 corePoolSize,maximumPoolSize,workQueue之间关系。 

        1、当线程池小于corePoolSize时,新提交任务将创建一个新线程执行任务,即使此时线程池中存在空闲线程。 

        2、当线程池达到corePoolSize时,新提交任务将被放入workQueue中,等待线程池中任务调度执行 

        3、当workQueue已满,且maximumPoolSize>corePoolSize时,新提交任务会创建新线程执行任务 

        4、当提交任务数超过maximumPoolSize时,新提交任务由RejectedExecutionHandler处理 (拒绝策略)

        5、当线程池中超过corePoolSize线程,空闲时间达到keepAliveTime时,关闭空闲线程 

        6、当设置allowCoreThreadTimeOut(true)时,线程池中corePoolSize线程空闲时间达到keepAliveTime也将关闭

四、创建线程池的方式

4.1 固定长度线程池:(指定工作线程数量的线程池)——newFixedThreadPool

        重用指定数目(nThreads)的线程,其背后使用 的是无界的工作队列,任何时候最多有 nThreads 个工作线程是活动的。这意味着,如果任务数量超过了活动队列数目,将在工作队列中等待空闲线程出现;如果有工作线程退出,将 会有新的工作线程被创建,以补足指定的数目 nThreads;

        适用场景:可用于Web服务瞬时削峰,但需注意长时间持续高峰情况造成的队列阻塞。

package com.aaa.day03_threadPool.demo1;

import java.util.concurrent.Executor;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;

/**
 * @author :caicai
 * @date :Created in 2022/7/20 13:31
 * @description:固定长度线程池
 * @description: 适用场景:可用于Web服务瞬时削峰,但需注意长时间持续高峰情况造成的队列阻塞。
 */
public class FixedThreadPoolDemo {
    public static void main(String[] args) {
        // 实例化一个固定长度为3的线程池
        ExecutorService executorService = Executors.newFixedThreadPool(3);
        for (int i = 0; i < 20; i++) {
            // 使用ExecutorService.execute启动线程
            executorService.execute(new Runnable() {
                @Override
                public void run() {
                    System.out.println("线程"+Thread.currentThread().getName()+"正在执行。。。");
                }
            });
        }
        // 关闭线程池
        executorService.shutdown();
    }
}

结果展示:

 4.2 单独线程池:(单线程化的线程池)——newSingleThreadPool

        它的特点在于工作线程数目被限制为 1,操作一个无界 的工作队列,所以它保证了所有任务的都是被顺序执行,最多会有一个任务处于活动状态, 并且不允许使用者改动线程池实例,因此可以避免其改变线程数目

        适用场景: 想让任务高速顺序执行时,可以使用

package com.aaa.day03_threadPool.demo1;

import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;

/**
 * @author :caicai
 * @date :Created in 2022/7/20 13:55
 * @description:单独线程池
 * @description:适用场景: 想让任务高速顺序执行时,可以使用
 * @description:他的特点在于工作线程数目被限制为1,操作一个无界的工作队列,所以他保证了所有任务的都是被顺序执行
 *      最多会有一个任务处于活动状态,并且步允许使用者改动线程池实例,因此可以避免其改变线程数目
 * @modified By:
 * @version:
 */
public class SingleThreadPool {
    public static void main(String[] args) {
        // 实例化对象
        ExecutorService executorService = Executors.newSingleThreadExecutor();
        for (int i = 0; i < 20; i++) {
            // 启动线程
            executorService.execute(new Runnable() {
                @Override
                public void run() {
                    System.out.println("线程"+Thread.currentThread().getName()+"正在执行。。。。");
                }
            });
        }
        // 关闭线程池
        executorService.shutdown();
    }
}

       结果展示

 4.3 缓存线程池(可缓存线程池)——newCachedThreadPool

        它会试图缓存线程并重用,当无缓存线程可用时,就会创建新的工作线程;如果 线程闲置的时间超过 60 秒,则被终止并移出缓存;长时间闲置时,这种线程池,不会消耗 什么资源。

        应用场景:它是一种用来处理大量短时间工作任务的线程池

package com.aaa.day03_threadPool.demo1;

import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;

/**
 * @author :caicai
 * @date :Created in 2022/7/20 13:45
 * @description:缓存线程池
 * @description:应用场景:它是一种用来处理大量短时间工作任务的线程池
 * @description:它会试图缓存线程并重用,当无缓存线程可用时,就会创建新的工作线程;
*        如果线程闲置的时间超过60秒,则被终止并移除缓存;长时间闲置时,这种线程池,不会消耗什么资源
 * @modified By:
 * @version:
 */
public class CacheThreadPool {
    public static void main(String[] args) {
//         byte 1Byte =8bit   0000 0000  -> 1111  1111   2^8 =256   -128  - 127
//         int  4byte =32bit                             2^32       -2^31 - 2^31
        ExecutorService executorService = Executors.newCachedThreadPool();
        for (int i = 0; i < 20; i++) {
            // 使用ExecutorService.execute启动线程
            executorService.execute(new Runnable() {
                @Override
                public void run() {
                    System.out.println("线程"+Thread.currentThread().getName()+"正在执行了。。。。");
                }
            });
        }
        // 关闭线程池
        executorService.shutdown();
    }
}

        结果展示

 4.4 可调度线程池(定时线程池)——newScheduledThreadPool

        可以进行定时或周期性的工作调度,区别在于单一工作线程还是多个工作线程

        应用场景:定时或周期性任务执行

package com.aaa.day03_threadPool.demo1;

import java.util.Date;
import java.util.concurrent.Executors;
import java.util.concurrent.ScheduledExecutorService;
import java.util.concurrent.TimeUnit;

/**
 * @author :caicai
 * @date :Created in 2022/7/20 14:08
 * @description:可调度线程池
 * @modified By:
 * @version:
 */
public class ScheduledThreadPool {
    public static void main(String[] args) {
        // 实例化线程池
        ScheduledExecutorService scheduledExecutorService = Executors.newScheduledThreadPool(3);

        // 让当前运行的线程延迟5秒后开始执行
        scheduledExecutorService.schedule(new Runnable() {
            @Override
            public void run() {
                System.out.println("线程"+Thread.currentThread().getName()+"正在执行。。。。");
            }
        },3, TimeUnit.SECONDS);


        // 让当前运行的线程延迟5秒后开始执行,并且以后每隔3秒执行一次  FixedRate 两次开始执行之间的间隔
//        scheduledExecutorService.scheduleAtFixedRate(new Runnable() {
//            @Override
//            public void run() {
//                System.out.println("线程"+Thread.currentThread().getName()+"开始执行。。。。"+new Date());
//            }
//        },2,3,TimeUnit.SECONDS);


        // 让当前运行的线程延迟2秒后开始执行,并且以后每隔3秒执行一次
        // FixedDelay  上次任务结束到下次任务开始间隔时间
        /*scheduledExecutorService.scheduleWithFixedDelay(new Runnable() {
            @Override
            public void run() {
                System.out.println("线程"+Thread.currentThread().getName()+"开始执行。。。"+new Date());
                try {
                    Thread.sleep(2000);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
                System.out.println("线程"+Thread.currentThread().getName()+"执行结果。。。"+new Date());
            }
        },2,3,TimeUnit.SECONDS);*/
    }
}

        测试结果:

 五、submit和execute区别

        本质区别是Runnable和 Callable区别

        Submit和Execute区别:

    1)execute 参数只能是Runnable的实现类    submit 参数既能是Runnable的实现类 可以是Callable的实现类

    2)execute  没有返回值  submit有返回值,并且可以通过  future 的get() 获取

    3)execute  无法处理异常  submit 可以处理异常,并且可以通过  future 的get() 打印异常堆栈信息

代码展示:

package com.aaa.day03_threadPool.demo2;

import java.util.UUID;
import java.util.concurrent.*;

/**
 * @author :caicai
 * @date :Created in 2022/7/20 14:34
 * @description:submit和execute区别
 * 1、execute  参数只能是Runnable的实现类  submit 参数既能是Runnable的实现类 也可以是Callable的实现类
 * 2、execute 没有返回值  submit有返回值,并且可以通过  future 的get() 获取
 * 3、execute 无法处理异常  submit 可以处理异常,并且
 */
public class SubmitAndExecute {
    public static void main(String[] args) throws ExecutionException, InterruptedException {
        // 实例化一个固定长度为3的线程池
        ExecutorService executorService = Executors.newFixedThreadPool(3);
        // 区别1
        executorService.execute(new Runnable() {
            @Override
            public void run() {
                System.out.println("参数只能是Runnable的实现类。。。");
                System.out.println("===============================================");
            }
        });
        executorService.submit(new Runnable() {
            @Override
            public void run() {
                System.out.println("submit的参数既能是Runnable的实现类。。。");
            }
        });
        executorService.submit(new Callable<Object>() {
            @Override
            public Object call() throws Exception {
                System.out.println("submit的参数还可以是Callable的实现类。。。。");
                return null;
            }
        });
        Future<?> future = executorService.submit(new Callable<String>() {
            @Override
            public String call() throws Exception {
                System.out.println("submit的参数还可以是Callable的实现类。。。");
                String randomStr = UUID.randomUUID().toString();
                System.out.println(randomStr);
                //System.out.println(1/0);
                return randomStr;
            }
        });
        System.out.println("获取执行结果:"+future.get());
        executorService.shutdown();

    }
}

结果展示:

  • 3
    点赞
  • 20
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
线程安全,并发的知识有加深认知;当然,现在用过的东西并不是代表以后还能娴熟的使用,做好笔记非常重要; 1:必须明白为什么要使用线程池:(这点很重要)   a:手上项目所需,因为项目主要的目的是实现多线程的数据推送;需要创建多线程的话,那就要处理好线程安全的问题;因为项目需要,还涉及到排队下载的功能,所以就选择了线程池来管理线程以及线程池里面的任务队列workQueue来实现项目所需的功能;   b:在实际使用中,服务器在创建和销毁线程上花费的时间和消耗的系统资源都相当大,甚至可能要比在处理实际的用户请求的时间和资源要多的多。除了创建和销毁线程的开销之外,活动的线程也需要消耗系统资源。如果在一个jvm里创建太多的线程,可能会使系统由于过度消耗内存或“切换过度”而导致系统资源不足。为了防止资源不足,服务器应用程序需要采取一些办法来限制任何给定时刻处理的请求数目,尽可能减少创建和销毁线程的次数,特别是一些资源耗费比较大的线程的创建和销毁,尽量利用已有对象来进行服务,这就是“池化资源”技术产生的原因。 线程池主要用来解决线程生命周期开销问题和资源不足问题(这段是摘自网络) 2:如何创建一个线程池:    复制代码 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; } 复制代码 这里只是创建线程池其中的一个构造函数;其实其他的构造函数最终还是调用的这个构造函数; 说明一下这些参数的作用: corePoolSize:核心池的大小,在创建了线程池后,线程池中的线程数为0,当有任务来之后,就会创建一个线程去执行任务,当线程池中的线程数目达到corePoolSize后,就会把到达的任务放到缓存队列当中; maximumPoolSize:线程池最大线程数,它表示在线程池中最多能创建多少个线程;这个参数是跟后面的阻塞队列联系紧密的;只有当阻塞队列满了,如果还有任务添加到线程池的话,会尝试new 一个Thread的进行救急处理,立马执行对应的runnable任务;如果继续添加任务到线程池,且线程池中的线程数已经达到了maximumPoolSize,那么线程就会就会执行reject操作(这里后面会提及到) keepAliveTime:表示线程没有任务执行时最多保持多久时间会终止;默认情况下,只有当线程池中的线程数大于corePoolSize时,keepAliveTime才会起作用;即当线程池中的线程数大于corePoolSize时,如果一个线程空闲的时间达到keepAliveTime,则会终止,直到线程池中的线程数不超过corePoolSize。但是如果调用了allowCoreThreadTimeOut(boolean)方法并设置了参数为true,在线程池中的线程数不大于corePoolSize时,keepAliveTime参数也会起作用,直到线程池中的阻塞队列大小为0;(这部分通过查看ThreadPoolExecutor的源码分析--getTask()部分); unit:参数keepAliveTime的时间单位,有7种取值,在TimeUnit类中有7种静态属性(时间单位) workQueue:一个阻塞队列,用来存储等待执行的任务,这个参数的选择也很重要,会对线程池的运行过程产生重大影响,一般来说,这里的阻塞队列有以下几种选择     ArrayBlockingQueue;   LinkedBlockingQueue;   SynchronousQueue;   ArrayBlockingQueue和PriorityBlockingQueue使用较少,一般使用LinkedBlockingQueue和Synchronous。线程池的排队策略与BlockingQueue有关。 threadFactory:线程工厂,主要用来创建线程:默认值 DefaultThreadFactory; handler:表示当拒绝处理任务时的策略,就是上面提及的reject操作;有以下四种取值:   ThreadPoolExecutor.AbortPolicy:丢弃任务并抛出RejectedExecutionException异常。(默认handle)   ThreadPoolExecutor.DiscardPolicy:也是丢弃任务,但是不抛出异常。   ThreadPoolExecutor.DiscardOldestPolicy:丢弃队列最前面的任务,然后重新尝试执行任务(重复此过程)   ThreadPoolExecutor.CallerRunsPolicy:由调用线程处理该任务 3:对线程池的基本使用及其部分源码的分析(注意:这里的源码分析是基于jdk1.6;) a:线程池的状态 volatile int runState; static final int RUNNING = 0; 运行状态 static final int SHUTDOWN = 1; 关闭状态;SHUTDOWN状态,此时线程池不能够接受新的任务,它会等待所有任务执行完毕 static final int STOP = 2;停止状态;此时线程池不能接受新的任务,并且会去尝试终止正在执行的任务 static final int TERMINATED = 3;终止状态;当线程池处于SHUTDOWN或STOP状态,并且所有工作线程已经销毁,任务缓存队列已经清空或执行结束后,线程池被设置为TERMINATED状态 b:参数再次说明。这是摘自网络的解释,我觉得他比喻的很好,所以这里直接就用它的解释   这里要重点解释一下corePoolSize、maximumPoolSize、largestPoolSize三个变量。   corePoolSize在很多地方被翻译成核心池大小,其实我的理解这个就是线程池的大小。举个简单的例子:   假如有一个工厂,工厂里面有10个工人,每个工人同时只能做一件任务。   因此只要当10个工人中有工人是空闲的,来了任务就分配给空闲的工人做;   当10个工人都有任务在做时,如果还来了任务,就把任务进行排队等待;   如果说新任务数目增长的速度远远大于工人做任务的速度,那么此时工厂主管可能会想补救措施,比如重新招4个临时工人进来;   然后就将任务也分配给这4个临时工人做;   如果说着14个工人做任务的速度还是不够,此时工厂主管可能就要考虑不再接收新的任务或者抛弃前面的一些任务了。   当这14个工人当中有人空闲时,而新任务增长的速度又比较缓慢,工厂主管可能就考虑辞掉4个临时工了,只保持原来的10个工人,毕竟请额外的工人是要花钱的。   这个例子中的corePoolSize就是10,而maximumPoolSize就是14(10+4)。   也就是说corePoolSize就是线程池大小,maximumPoolSize在我看来是线程池的一种补救措施,即任务量突然过大时的一种补救措施。   不过为了方便理解,在本文后面还是将corePoolSize翻译成核心池大小。   largestPoolSize只是一个用来起记录作用的变量,用来记录线程池中曾经有过的最大线程数目,跟线程池的容量没有任何关系。 c:添加线程池任务的入口就是execute(); 复制代码 public void execute(Runnable command) { if (command == null) throw new NullPointerException();//任务为空时抛出异常 //如果线程池线程大小小于核心线程,就新建一个线程加入任务并启动线程 //如果线程池线程大小大于核心线且且添加任务到线程失败,就把任务添加到阻塞队列 if (poolSize >= corePoolSize || !addIfUnderCorePoolSize(command)) {//新建线程并启动 if (runState == RUNNING && workQueue.offer(command)) {//添加任务到队列 if (runState != RUNNING || poolSize == 0) ensureQueuedTaskHandled(command);//添加到队列失败或已满,做拒接任务处理策略 } //若阻塞队列失败或已满;这里新建一个线程并启动做应急处理(这里就是用到了maximumPoolSize参数) else if (!addIfUnderMaximumPoolSize(command)) reject(command); // 若线程池的线程超过了maximumPoolSize;就做拒绝处理任务策略 } } 复制代码 -->>继续跟踪代码到addIfUnderCorePoolSize(Runnable firstTask):函数名称就可以看出来这个函数要执行的什么;如果线程池的线程小于核心线程数corePoolSize就新建线程加入任务并启动线程【在今后的开发中尽量把需要做的功能在函数名体现出来】 复制代码 private boolean addIfUnderCorePoolSize(Runnable firstTask) { Thread t = null; final ReentrantLock mainLock = this.mainLock;//获取当前线程池的锁 mainLock.lock();//加锁 try { /* 这里线程池线程大小还需要判断一次;前面的判断过程中并没有加锁,因此可能在execute方法判断的时候poolSize小于corePoolSize,而判断完之后,在其他线程中又向线程池提交了任务,就可能导致poolSize不小于corePoolSize了,所以需要在这个地方继续判断 */ if (poolSize < corePoolSize && runState == RUNNING) t = addThread(firstTask);//新建线程 } finally { mainLock.unlock(); } if (t == null) return false; t.start();//若创建线程超过,就启动线程池的线程 return true; } private Thread addThread(Runnable firstTask) { Worker w = new Worker(firstTask);//worker:ThreadPoolExecutor的内部类; Thread t = threadFactory.newThread(w);//使用线程工厂创建一个线程 if (t != null) { w.thread = t; workers.add(w);//保存线程池正在运行的线程 int nt = ++poolSize;//线程池的线程数加1 if (nt > largestPoolSize) largestPoolSize = nt; } return t; } 复制代码 -->>接下来定位worker类,看看线程池里的线程是如何执行的 上面的addIfUnderCorePoolSize(..)已经把线程启动了;现在就直接查看worker 的run()方法了 复制代码 public void run() { try { Runnable task = firstTask;//该线程的第一个任务,执行完后就从阻塞队列取任务执行 firstTask = null; while (task != null || (task = getTask()) != null) {//getTask()从队列去任务执行 runTask(task);//线程执行任务 task = null; } } finally { workerDone(this);//若任务全部执行完,就开始尝试去停止线程池;这部分代码就不再追踪下去,有兴趣的读者可以自己打开源码分析,不必害怕,学习大神们的编码方式,看源码能让你学习到很多 } } private void runTask(Runnable task) { final ReentrantLock runLock = this.runLock; runLock.lock(); try { //多次检查线程池有没有关闭 if (runState < STOP && Thread.interrupted() && runState >= STOP) thread.interrupt(); boolean ran = false; //这里就可以继承ThreadPoolExecutor,并覆盖beforeExecute(...)该方法,来做一些执行任务之前的统计工作或者用来保存正在执行的任务 beforeExecute(thread, task); try { task.run(); ran = true; //这里就可以继承ThreadPoolExecutor,并覆盖beforeExecute(...)该方法,来做一些执行任务完成之后的统计工作或者用来保存正在执行的任务 afterExecute(task, null); ++completedTasks;//统计总共执行的任务数 } catch (RuntimeException ex) { if (!ran) afterExecute(task, ex); throw ex; } } finally { runLock.unlock(); } } 复制代码 至此线程池基本的流程完了; 再说说我在项目中的使用: MyExtendThreadPoolExecutor 继承了 ThreadPoolExecutor,并覆盖了其中的一些方法 复制代码 public class MyExtendThreadPoolExecutor extends ThreadPoolExecutor{ public static Logger logger=LoggerFactory.getLogger(MyExtendThreadPoolExecutor.class); /** * 记录运行中任务 */ private LinkedBlockingQueue<Runnable> workBlockingQueue=new LinkedBlockingQueue<Runnable>(); public MyExtendThreadPoolExecutor(int corePoolSize, int maximumPoolSize, long keepAliveTime, TimeUnit unit, BlockingQueue<Runnable> workQueue) { super(corePoolSize, maximumPoolSize, keepAliveTime, unit, workQueue); } @Override protected void beforeExecute(Thread t, Runnable r) { super.beforeExecute(t, r); workBlockingQueue.add((GtdataBreakpointResumeDownloadThread)r);//保存在运行的任务 logger.info("Before the task execution"); } @Override protected void afterExecute(Runnable r, Throwable t) { super.afterExecute(r, t); workBlockingQueue.remove((GtdataBreakpointResumeDownloadThread)r);//移除关闭的任务 logger.info("After the task execution"); } /** * * Description: 正在运行的任务 * @return LinkedBlockingQueue<Runnable><br> * @author lishun */ public LinkedBlockingQueue<Runnable> getWorkBlockingQueue() { return workBlockingQueue; } } 复制代码 MyExtendThreadPoolExecutor pool = new MyExtendThreadPoolExecutor(3, 3,60L,TimeUnit.SECONDS,new LinkedBlockingQueue <Runnable>()); //创建线程池 复制代码 public void addToThreadPool(DownloadRecord downloadRecord){ BlockingQueue<Runnable> waitThreadQueue = pool.getQueue();//Returns the task queue LinkedBlockingQueue<Runnable> workThreadQueue =pool.getWorkBlockingQueue();//Returns the running work GtdataBreakpointResumeDownloadThread downloadThread = new GtdataBreakpointResumeDownloadThread(downloadRecord);//需要执行的任务线程 if (!waitThreadQueue.contains(downloadThread)&&!workThreadQueue.contains(downloadThread)) {//判断任务是否存在正在运行的线程或存在阻塞队列,不存在的就加入线程池(这里的比较要重写equals()) Timestamp recordtime = new Timestamp(System.currentTimeMillis()); logger.info("a_workThread:recordId="+downloadRecord.getId()+",name="+downloadRecord.getName()+" add to workThreadQueue"); downloadThread.setName("th_"+downloadRecord.getName()); pool.execute(downloadThread);//添加到线程池 }else{ logger.info("i_workThread:recordId="+downloadRecord.getId()+",name="+downloadRecord.getName()+" in waitThreadQueue or workThreadQueue"); } }

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

菜鸡本蔡

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值