java手写队列_Java手写线程池:带你翻过线程池这座山

如果想玩转 Java 的多线程与高并发,线程池是你永远也绕不过的山。既然绕不过,我们就啃他,吃透线程池,玩转高并发。

Jdk里的线程池 主要属性 private volatile int corePoolSize; //核心线程数private volatile int maximumPoolSize; //最大线程数private volatile long keepAliveTime; //存活时间private final BlockingQueue workQueue; //任务等待队列private volatile ThreadFactory threadFactory; //线程工厂private volatile RejectedExecutionHandler handler; //拒绝策略

下面介绍一下线程池执行任务的流程,理解各个属性的意义。当一个线程池初始化,向线程池提交任务,线程池新建线程执行任务,随着线程创建,线程数逐渐增多,当达到 corePoolSize 线程池将不再新建线程,而是将任务放入任务等待队列 workQueue 。再持续向线程池提交任务,当等待队列满了,这时会继续新建线程,直到到达最大线程数 maximumPoolSize,如果还继续有任务到来,线程池无法处理,这时就启动拒绝策略。

这个过程我们可以以生活中的例子比喻一下。大致我们把线程池理解为理发店。那么流程就是:来了顾客开始理发,比如只有4个理发师4个座位,相当于核心线程。那么来了过多的顾客,理发师忙不过来就会先让你去等候区稍等排队等待,前面有理完发的会叫你,相当于等待队列。等待区满了呢?现实中理发店肯定不会拒绝顾客的啊,他可能让你先在外面等。但如果等待区每天都爆满,那么老板可能会考虑扩大店面,扩充理发师团队了。所以,这只是个大致的比喻。

自己手写线程池

手写线程池,代码:

public class MyThreadPoolV3 { /** * 核心线程数(理发师数量) */ private volatile int corePoolSize; /** * 任务等待队列(等待区座位数) */ private final BlockingQueue workQueue; /** * 线程容器(理发师作业区) */ private List workers; public MyThreadPoolV3(int corePoolSize, BlockingQueue workQueue) { this.corePoolSize = corePoolSize; this.workQueue = workQueue; workers = Collections.synchronizedList(new ArrayList<>(corePoolSize)); for (int i = 0; i < corePoolSize; i++) { Worker worker = new Worker(); workers.add(worker); worker.start(); } } /** * 将任务加到队列(给顾客安排位置) * * @param task * @return */ public boolean submit(Runnable task) { return workQueue.offer(task); } // ---------------- /** * 线程(理发师) */ private class Worker extends Thread { @Override public void run() { while (true) { Runnable task = null; try { //从任务队列取任务,如空则阻塞等待。(去等候区叫顾客,如果没有则原地等待一会) task = workQueue.take(); } catch (InterruptedException e) { e.printStackTrace(); } if (task != null) { //取到任务开始工作(开始理发) task.run(); } } } }}

测试用例代码:

public class ThreadPoolDemoV3 { private static LinkedBlockingQueue workQueue = new LinkedBlockingQueue<>(30); private static MyThreadPoolV3 threadPoolV3 = new MyThreadPoolV3(3, workQueue); public static void main(String[] args) { AtomicInteger atomicInteger = new AtomicInteger(0); Runnable task = new Runnable() { @Override public void run() { System.out.println(atomicInteger.incrementAndGet() + "号顾客来理发,为其理发的理发师是:" + Thread.currentThread().getName()); } }; for (int i = 0; i < 30; i++) { threadPoolV3.submit(task); } }}

测试结果:

3079255fa225261d5aca2d9fa3dbb6a7.png

我们发现无论来多少顾客理发,始终是0、1、2理发师轮流工作。这就是线程池的作用,实现了线程的复用,节省了资源,提升了程序性能。

当然,这个线程池只是个简版,让我们理解线程池的流程。和jdk的线程池比还有很多功能没有完善,等待队列满了将新建线程执行,直到达到最大线程数;线程的存活时间;拒绝策略等。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值