加入收藏 | 设为首页 | 会员中心 | 我要投稿 武汉站长网 (https://www.027zz.cn/)- 科技、建站、经验、云计算、5G、大数据,站长网!
当前位置: 首页 > 综合聚焦 > 移动互联 > 评测 > 正文

3年工作经验,工作中还不会使用多线程?阿里P6:别慌,我都总结好了

发布时间:2019-07-18 21:56:10 所属栏目:评测 来源:Jay_huaxiao
导读:副标题#e# 掌握线程池是后端程序员的基本要求,相信大家求职面试过程中,几乎都会被问到有关于线程池的问题。我在网上搜集了几道经典的线程池面试题,并以此为切入点,谈谈我对线程池的理解。如果有哪里理解不正确,非常希望大家指出,接下来大家一起分析学

通过以上分析,submit执行的任务,可以通过Future对象的get方法接收抛出的异常,再进行处理。 我们再通过一个demo,看一下Future对象的get方法处理异常的姿势,如下图:

3年工作经验,工作中还不会使用多线程?阿里P6:别慌,我都总结好了

其他两种处理线程池异常方案

除了以上1.在任务代码try/catch捕获异常,2.通过Future对象的get方法接收抛出的异常,再处理两种方案外,还有以上两种方案:

3.为工作者线程设置UncaughtExceptionHandler,在uncaughtException方法中处理异常

我们直接看这样实现的正确姿势:

  1. ExecutorService threadPool = Executors.newFixedThreadPool(1, r -> { 
  2.  Thread t = new Thread(r); 
  3.  t.setUncaughtExceptionHandler( 
  4.  (t1, e) -> { 
  5.  System.out.println(t1.getName() + "线程抛出的异常"+e); 
  6.  }); 
  7.  return t; 
  8.  }); 
  9.  threadPool.execute(()->{ 
  10.  Object object = null; 
  11.  System.out.print("result## " + object.toString()); 
  12.  }); 
  13. 复制代码 

运行结果:

3年工作经验,工作中还不会使用多线程?阿里P6:别慌,我都总结好了

4.重写ThreadPoolExecutor的afterExecute方法,处理传递的异常引用

这是jdk文档的一个demo:

  1. class ExtendedExecutor extends ThreadPoolExecutor { 
  2.  // 这可是jdk文档里面给的例子。。 
  3.  protected void afterExecute(Runnable r, Throwable t) { 
  4.  super.afterExecute(r, t); 
  5.  if (t == null && r instanceof Future<?>) { 
  6.  try { 
  7.  Object result = ((Future<?>) r).get(); 
  8.  } catch (CancellationException ce) { 
  9.  t = ce; 
  10.  } catch (ExecutionException ee) { 
  11.  t = ee.getCause(); 
  12.  } catch (InterruptedException ie) { 
  13.  Thread.currentThread().interrupt(); // ignore/reset 
  14.  } 
  15.  } 
  16.  if (t != null) 
  17.  System.out.println(t); 
  18.  } 
  19. }} 
  20. 复制代码 

因此,被问到线程池异常处理,如何回答?

3年工作经验,工作中还不会使用多线程?阿里P6:别慌,我都总结好了

。线程池的工作队列

线程池都有哪几种工作队列?

  • ArrayBlockingQueue
  • LinkedBlockingQueue
  • DelayQueue
  • PriorityBlockingQueue
  • SynchronousQueue

ArrayBlockingQueue

ArrayBlockingQueue(有界队列)是一个用数组实现的有界阻塞队列,按FIFO排序量。

LinkedBlockingQueue

LinkedBlockingQueue(可设置容量队列)基于链表结构的阻塞队列,按FIFO排序任务,容量可以选择进行设置,不设置的话,将是一个无边界的阻塞队列,最大长度为Integer.MAX_VALUE,吞吐量通常要高于ArrayBlockingQuene;newFixedThreadPool线程池使用了这个队列

DelayQueue

DelayQueue(延迟队列)是一个任务定时周期的延迟执行的队列。根据指定的执行时间从小到大排序,否则根据插入到队列的先后排序。newScheduledThreadPool线程池使用了这个队列。

PriorityBlockingQueue

PriorityBlockingQueue(优先级队列)是具有优先级的无界阻塞队列;

SynchronousQueue

SynchronousQueue(同步队列)一个不存储元素的阻塞队列,每个插入操作必须等到另一个线程调用移除操作,否则插入操作一直处于阻塞状态,吞吐量通常要高于LinkedBlockingQuene,newCachedThreadPool线程池使用了这个队列。

针对面试题:线程池都有哪几种工作队列? 我觉得,回答以上几种ArrayBlockingQueue,LinkedBlockingQueue,SynchronousQueue等,说出它们的特点,并结合使用到对应队列的常用线程池(如newFixedThreadPool线程池使用LinkedBlockingQueue),进行展开阐述, 就可以啦。

几种常用的线程池

  • newFixedThreadPool (固定数目线程的线程池)
  • newCachedThreadPool(可缓存线程的线程池)
  • newSingleThreadExecutor(单线程的线程池)
  • newScheduledThreadPool(定时及周期执行的线程池)

newFixedThreadPool

  1.  public static ExecutorService newFixedThreadPool(int nThreads, ThreadFactory threadFactory) { 
  2.  return new ThreadPoolExecutor(nThreads, nThreads, 
  3.  0L, TimeUnit.MILLISECONDS, 
  4.  new LinkedBlockingQueue<Runnable>(), 
  5.  threadFactory); 
  6.  } 
  7. 复制代码 

线程池特点:

  • 核心线程数和最大线程数大小一样
  • 没有所谓的非空闲时间,即keepAliveTime为0
  • 阻塞队列为无界队列LinkedBlockingQueue

(编辑:武汉站长网)

【声明】本站内容均来自网络,其相关言论仅代表作者个人观点,不代表本站立场。若无意侵犯到您的权利,请及时与联系站长删除相关内容!

热点阅读