java并发基础(五)--- 线程池的使用
第8章介绍的是线程池的使用,直接进入正题。
一、线程饥饿死锁和饱和策略
1.线程饥饿死锁
在线程池中,如果任务依赖其他任务,那么可能产生死锁。举个极端的例子,在单线程的Executor中,如果一个任务提交了另一个任务到相同的Executor中,并等待其返回,那么就会发生死锁。第二个任务停留在工作队列中,第一个又一直等待(因为是单线程)。这块记住一个信息,就是如果线程池中的任务是互相依赖的,除非线程池无限大,否则就有可能产生线程饥饿死锁,而且是否产生死锁要看时机,这也就是为什么Executor框架提供的实现中提倡使用newCachedThreadPool作为默认实现,原因之一就是它的线程数无限大(当然是理论上)。
2.饱和策略
当线程池的有界队列填满后,该用一种什么样的策略来处理没能添加进来的任务,JDK提供了几种默认实现。
(1)中止(Abort):默认策略,抛出未检出的RejectedExecutionException。
(2)抛弃(Discard):新提交的任务无法保存到队列中,则被抛弃。
(3)抛弃最旧的(Discard-Oldest):抛弃下一个将被执行的任务,然后尝试提交新的任务。这个策略不适合优先队列,因为会抛弃优先级最高的任务。
(4)调用者运行(Caller-Runs):该策略不会抛弃任务,也不会抛出异常,而是将某些任务回退到调用者,从而降低新任务的流量。
ThreadPoolExecutor的饱和策略通过调用setRejectedExecutionHandler来修改。
二、示例:搬箱子
搬箱子计算从初始位置到目标位置的所有合法移动,以及每次移动的结果位置,感觉上有点像阿尔法狗下围棋的逻辑类似,当然不是一个层面的问题,我们只是用这个例子熟悉下线程池的使用。这块涉及到的比如如何判断当前位置是否是目标位置,以及如何计算所有合法的移动等等,我们先不管,用接口代替,毕竟我们要思考的是线程池的使用。
首先是搬箱子的抽象类,该抽象类应该包括这么几个接口:1.初始化位置 2.判断当前位置是否是目标位置 3.列出所有可能的移动 4.执行移动
//P:位置类 M:移动类
public interface Puzzle<P,M>{
//初始化位置
P initialPosition();
//判断该位置是否是目标位置
boolean isGoal(P position);
//列出从position开始的所有合法移动
Set<M> legalMoves(P position);
//从指定位置开始移动 返回移动后的结果位置
P move(P position,M move);
}
这个接口可以解决问题,找到合法移动,然后执行移动,接下来,就是如何操作了,我们先看串行代码如何写:
public class SequentialPuzzleSolver<P,M>{
private final Puzzle<P,M> puzzle;
//所有移动位置的集合
private final Set<P> seen = new HashSet<P>();
public SequentialPuzzleSolver(Puzzle<P, M> puzzle) {
super();
this.puzzle = puzzle;
}
public List<M> solve(){
P pos = puzzle.initialPosition();
return search(new Node<P,M>(pos, null, null));
}
private List<M> search(Node<P,M> node){
if (!seen.contains(node.pos)) {
seen.add(node.pos);
if (puzzle.isGoal(node.pos)) {
return node.asMoveList();
}
for (M move:puzzle.legalMoves(node.pos)) {
//向指定位置移动返回最新位置
P pos = puzzle.move(node.pos, move);
//将最新位置封装成node继续移动
Node<P,M> child = new Node<P,M>(pos, move, node);
//递归
List<M> result = search(child);
if (result != null) {
return result;
}
}
}
return null;
}
static class Node<P,M>{
final P pos;
final M move;
final Node<P,M> prev;
Node(P pos, M move, Node<P, M> prev) {
super();
this.pos = pos;
this.move = move;
this.prev = prev;
}
List<M> asMoveList(){
List<M> solution = new LinkedList<M>();//用链表,增删快
for (Node<P,M> n = this;n.move != null;n=n.prev) {
solution.add(0,n.move);//最新一次的移动下标为0
}
return solution;
}
}
}
Node是对Positon的进一步封装,保存了当前node的位置position和移动move以及前一个节点。这样不断追溯就可以得到完整的移动轨迹。可以看到,串行的思路是先得到所有可能的移动,然后遍历,一个一个移动,每移动一次再查找当前位置的可能移动,再遍历......也就是循环递归调用,这种显然是没有效率的,可以并发的地方也在这里。这里必须明确任务的边界即:一次移动。
//并发处理
public class ConcurrentPuzzleSolver<P,M>{
private final Puzzle<P,M> puzzle;
private final ExecutorService exec;
private final ConcurrentHashMap<P, Boolean> seen; final ValueLatch<Node<P,M>> solution = new ValueLatch<Node<P,M>>(); public ConcurrentPuzzleSolver(Puzzle<P, M> puzzle, ExecutorService exec,
ConcurrentHashMap<P, Boolean> seen) {
super();
this.puzzle = puzzle;
this.exec = exec;
this.seen = seen;
} public List<M> solve() throws InterruptedException{
try {
P p = puzzle.initialPosition();
exec.execute(newTask(p,null,null));
//阻塞直到找到答案
Node<P,M> solnNode = solution.getValue();
return (solnNode == null)?null:solnNode.asMoveList();
} catch (Exception e) {
exec.shutdown();
}
} protected Runnable newTask(P p,M m,Node<P,M> n){
return new SolverTask(p,m,n);
} class SolverTask extends Node<P,M> implements Runnable {
public SolverTask(P pos, M move, Node<P, M> prev) {
super(pos, move, prev);
} public void run() {
//首先访问闭锁,如果有答案则停止
if (solution.isSet()||seen.putIfAbsent(pos, true) != null) {
return;
}
if (puzzle.isGoal(pos)) {
solution.setValue(this);
}else {
for (M m:puzzle.legalMoves(pos)) {
exec.execute(newTask(puzzle.move(pos, m), m, this));
}
}
}
}
}
//有答案后停止 闭锁实现
public class ValueLatch<T>{
private T value = null;
private final CountDownLatch done = new CountDownLatch(1); public boolean isSet(){
return (done.getCount() == 0);
} public synchronized void setValue(T newValue){
if (!isSet()) {
value = newValue;
done.countDown();
}
} public T getValue() throws InterruptedException{
done.await();
synchronized (this) {
return value;
}
}
}
ValueLatch的作用是当线程池找到一个答案后停止其他任务,组合CountDownLatch实现,这是闭锁的另一个例子。第一个例子在java并发基础(二)的第三部分同步容器中介绍过了。在获得第一个答案之前,主线程将一直等待,ValueLatch中的getValue将一直阻塞,直到有线程设置了这个值。找到第一个答案后关闭线程池,不再接受新的任务,另外,为了避免抛出RejectedExecutionException,设置线程池饱和策略为Discard。
java并发基础(五)--- 线程池的使用的更多相关文章
- Java并发编程:线程池的使用
Java并发编程:线程池的使用 在前面的文章中,我们使用线程的时候就去创建一个线程,这样实现起来非常简便,但是就会有一个问题: 如果并发的线程数量很多,并且每个线程都是执行一个时间很短的任务就结束了, ...
- Java并发编程:线程池的使用(转)
Java并发编程:线程池的使用 在前面的文章中,我们使用线程的时候就去创建一个线程,这样实现起来非常简便,但是就会有一个问题: 如果并发的线程数量很多,并且每个线程都是执行一个时间很短的任务就结束了, ...
- (转)Java并发编程:线程池的使用
背景:线程池在面试时候经常遇到,反复出现的问题就是理解不深入,不能做到游刃有余.所以这篇博客是要深入总结线程池的使用. ThreadPoolExecutor的继承关系 线程池的原理 1.线程池状态(4 ...
- Java并发编程:线程池的使用(转载)
转载自:https://www.cnblogs.com/dolphin0520/p/3932921.html Java并发编程:线程池的使用 在前面的文章中,我们使用线程的时候就去创建一个线程,这样实 ...
- Java并发编程:线程池的使用(转载)
文章出处:http://www.cnblogs.com/dolphin0520/p/3932921.html Java并发编程:线程池的使用 在前面的文章中,我们使用线程的时候就去创建一个线程,这样实 ...
- [转]Java并发编程:线程池的使用
Java并发编程:线程池的使用 在前面的文章中,我们使用线程的时候就去创建一个线程,这样实现起来非常简便,但是就会有一个问题: 如果并发的线程数量很多,并且每个线程都是执行一个时间很短的任务就结束了, ...
- 【转】Java并发编程:线程池的使用
Java并发编程:线程池的使用 在前面的文章中,我们使用线程的时候就去创建一个线程,这样实现起来非常简便,但是就会有一个问题: 如果并发的线程数量很多,并且每个线程都是执行一个时间很短的任务就结束了, ...
- 13、Java并发编程:线程池的使用
Java并发编程:线程池的使用 在前面的文章中,我们使用线程的时候就去创建一个线程,这样实现起来非常简便,但是就会有一个问题: 如果并发的线程数量很多,并且每个线程都是执行一个时间很短的任务就结束了, ...
- Java并发编程之线程池及示例
1.Executor 线程池顶级接口.定义方法,void execute(Runnable).方法是用于处理任务的一个服务方法.调用者提供Runnable 接口的实现,线程池通过线程执行这个 Runn ...
- Java并发编程:线程池ThreadPoolExecutor
多线程的程序的确能发挥多核处理器的性能.虽然与进程相比,线程轻量化了很多,但是其创建和关闭同样需要花费时间.而且线程多了以后,也会抢占内存资源.如果不对线程加以管理的话,是一个非常大的隐患.而线程池的 ...
随机推荐
- [转载]HTML5浏览器测试网站汇总
http://www.cnblogs.com/javawebsoa/archive/2012/04/19/2458224.html 浏览器支持情况统计 When Can IUse:图表经常更新,展示了 ...
- [整理]javascript压缩、格式化
1.使用packer来压缩JS文件 packer工具在线版:http://dean.edwards.name/packer/ 通过packer对js打包压缩的同时,执行Base62 encode编码后 ...
- 【LibreOJ】#6354. 「CodePlus 2018 4 月赛」最短路 异或优化建图+Dijkstra
[题目]#6354. 「CodePlus 2018 4 月赛」最短路 [题意]给定n个点,m条带权有向边,任意两个点i和j还可以花费(i xor j)*C到达(C是给定的常数),求A到B的最短距离.\ ...
- 乘法逆元(P3811)(四种方法)
适合单个的,费马小定理,exgcd,都是不错的选择,利用积性函数的方法和欧拉筛的方法适合批量求,但是论时间和空间的话,还是积性函数的方法比较好用,线性的. 题目链接:https://www.luogu ...
- 【转】深入理解C++中public、protected及private用法
首先明白以下两点: 1.类的一个特征就是封装,public和private作用就是实现这一目的. 即:用户代码(类外)可以访问public成员而不能访问private成员:private成员只能由类成 ...
- tomcat报错catalina.sh: line 401: /usr/java/jdk1.7.52/bin/java: No such file or directory
将生产服务器的Tomcat目录打包过来后解压后,启动Tomcat后,发现如下问题: # ./shutdown.sh Using CATALINA_BASE: /usr/local/tomcat ...
- jexus http to https
一.摘要 1.80端口上只要没有网站使用 hosts=* 这样的配置,jexus会自动将域名跳转到 https 上.也就是说,你把网站默认配置文件default中的hosts=*改成具体的其它的域名, ...
- 前后端分离之mockjs实战demo
基于vue-cli+webpack的demo 项目结构 axios文件夹用来创建axios相关配置: import axios from 'axios' import vue from 'vue' a ...
- tenaorflow函数(1)
TensorFlow 将图形定义转换成分布式执行的操作, 以充分利用可用的计算资源(如 CPU 或 GPU.一般你不需要显式指定使用 CPU 还是 GPU, TensorFlow 能自动检测.如果检测 ...
- SOA 解惑
SOA 解惑 SOA 不是一种技术,它是一种设计方法.最近一段时间我碰到了很多关于 SOA 的具有误导性的文章.尤其是,有些人混淆了 SOA 和诸如 BPM.ESB 以及复合事件处理 (CEP) 之类 ...