Java Thread系列(九)Master-Worker模式
Java Thread系列(九)Master-Worker模式
Master-Worker模式是常用的并行设计模式.
一、Master-Worker 模式核心思想
Master-Worker 系统由两个角色组成,Master 和 Worker,Master 负责接收和分配任务,Worker 负责处理子任务。任务处理过程中,Master 还负责监督任务进展和 Worker 的健康状态;Master 将接收 Client 提交的任务,并将任务的进展汇总反馈给 Client。各角色关系如下图:
二、Master-Worker 实现
(1) Master
import java.util.HashMap;
import java.util.Map;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.ConcurrentLinkedDeque;
public class Master {
//1. 应该有一个容器存放任务列表,这个容器需要支持高并发操作
private ConcurrentLinkedDeque<Task> taskQueue = new ConcurrentLinkedDeque<Task>();
//2. 应该有一个容器存放worker
private HashMap<String, Thread> workers = new HashMap<String, Thread>();
//3. 应该有一个容器存放结果集,这个容器需要支持高并发操作
private ConcurrentHashMap<String, Object> resultMap = new ConcurrentHashMap<String, Object>();
//4. 构造函数
public Master (Worker worker, int threadCount) {
//将任务列表和结果集传递给worker
worker.setTaskQueue(taskQueue);
worker.setResultMap(resultMap);
//初始化worder列表
for (int i = 0; i < threadCount; i++) {
workers.put("worker-" + i, new Thread(worker));
}
}
public Master (Worker worker) {
this(worker, Runtime.getRuntime().availableProcessors());
}
//5. 提交任务
public void submit (Task task) {
taskQueue.add(task);
}
//6. 执行方法 开启所有的线程
public void execute () {
for(Map.Entry<String, Thread> me : workers.entrySet()) {
me.getValue().start();
}
}
//7. 判断是否执行完毕
public boolean isComplete () {
for(Map.Entry<String, Thread> me : workers.entrySet()) {
if (me.getValue().getState() != Thread.State.TERMINATED)
return false;
}
return true;
}
//8. 处理结果集
public int getResult () {
int ret = 0;
for(Map.Entry<String, Object> me : resultMap.entrySet()) {
ret += (int)me.getValue();
}
return ret;
}
}
(2) Worker
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.ConcurrentLinkedDeque;
public class Worker implements Runnable {
private ConcurrentLinkedDeque<Task> taskQueue;
private ConcurrentHashMap<String, Object> resultMap;
@Override
public void run() {
while (true) {
Task task = taskQueue.poll();
if (task == null) break;
try {
Thread.sleep(50);
} catch (InterruptedException e) {
e.printStackTrace();
}
//返回结果集
resultMap.put(Integer.toString(task.getId()), handle(task));
}
}
private Object handle(Task task) {
return task.getCount();
}
public void setTaskQueue(ConcurrentLinkedDeque<Task> taskQueue) {
this.taskQueue = taskQueue;
}
public void setResultMap(ConcurrentHashMap<String, Object> resultMap) {
this.resultMap = resultMap;
}
}
(3) Task
public class Task {
private int id;
private String name;
private int count;
public Task() {}
public Task(int id, String name, int count) {
this.id = id;
this.name = name;
this.count = count;
}
public int getId() {
return id;
}
public void setId(int id) {
this.id = id;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public int getCount() {
return count;
}
public void setCount(int count) {
this.count = count;
}
@Override
public String toString() {
return "Task{" + "id=" + id + ", name='" + name + '\'' +
", count=" + count + '}';
}
}
(4) 测试
Master master = new Master(new Worker(), 1);
for (int i = 1; i <= 100; i++) {
master.submit(new Task(i, "task-" + i ,i));
}
master.execute();
long t1 = System.currentTimeMillis();
while (true) {
if (master.isComplete()) {
long t = System.currentTimeMillis() - t1;
System.out.printf("执行结果:%s;执行时间:%s", master.getResult(), t);
break;
}
}
每天用心记录一点点。内容也许不重要,但习惯很重要!
Java Thread系列(九)Master-Worker模式的更多相关文章
- Java Thread系列(十)Future 模式
Java Thread系列(十)Future 模式 Future 模式适合在处理很耗时的业务逻辑时进行使用,可以有效的减少系统的响应时间,提高系统的吞吐量. 一.Future 模式核心思想 如下的请求 ...
- Java Thread系列(十)生产者消费者模式
Java Thread系列(十)生产者消费者模式 生产者消费者问题(producer-consumer problem),是一个多线程同步问题的经典案例.该问题描述了两个共亨固定大小缓冲区的线程-即所 ...
- Java Thread系列(七)死锁
Java Thread系列(七)死锁 当线程需要同时持有多个锁时,有可能产生死锁.考虑如下情形: 线程 A 当前持有互斥所锁 lock1,线程 B 当前持有互斥锁 lock2.接下来,当线程 A 仍然 ...
- java多线程系列(九)---ArrayBlockingQueue源码分析
java多线程系列(九)---ArrayBlockingQueue源码分析 目录 认识cpu.核心与线程 java多线程系列(一)之java多线程技能 java多线程系列(二)之对象变量的并发访问 j ...
- Java Thread系列(四)线程通信
Java Thread系列(四)线程通信 一.传统通信 public static void main(String[] args) { //volatile实现两个线程间数据可见性 private ...
- Java Thread系列(五)synchronized
Java Thread系列(五)synchronized synchronized锁重入 关键字 synchronized 拥有锁重入的功能,也就是在使用 synchronized 时,当线程等到一个 ...
- Java Thread系列(三)线程安全
Java Thread系列(三)线程安全 一.什么是线程安全 线程安全概念:当多个线程访问某一个类(对象或方法)时,这个类始终都能表现出正确的行为,那么这个类(对象或方法)就是线程安全的. 线程安全来 ...
- Java Thread系列(二)线程状态
Java Thread系列(二)线程状态 一.线程的五种状态 新建状态(New):新创建了一个线程对象,尚未启动. 就绪状态(Runnable):也叫可运行状态.线程对象创建后,其他线程调用了该对象的 ...
- Java Thread系列(一)线程创建
Java Thread系列(一)线程创建 Java 中创建线程主要有三种方式:继承 Thread.实现 Runnable 接口.使用 ExecutorService.Callable.Future 实 ...
随机推荐
- 如何在CentOS7上安装MySQL并实现远程访问
传送门 本人乃学生小白一枚,近期在学习Linux,所以就简单记录一下~ 安装MySQL 首先,需要检查一下是否已经安装了MySQL : # rpm -qa | grep mysql 这时候没有任何输出 ...
- memsql filesystem pipeline 试用
一些功能类似drill ,比如s3,file ... 创建file pipeline 准备file mkdir -p /opt/db/ touch books.txt 内容如下: The Catche ...
- HDU1735 字数统计
版权声明:长风原创 https://blog.csdn.net/u012846486/article/details/28011667 字数统计 Time Limit: 1000/2000 MS (J ...
- vs2013 乱码问题
单击一下代码框 然后点文件 有个高级保存选项 改成utf-8 就可以显示中文了
- FastAdmin 前台会员分组的权限分析
FastAdmin 前台会员分组的权限分析 之前有用户反馈前台用户组权限无法选中无子节点的节点 1. 其实这个理解是不对的,是无法选中菜单可视的节点. 从这里 UserRule.php 可能可以看出. ...
- .NET泛型解析(上)
[1]:泛型介绍 泛型是C#2.0中一个重要的新特性,泛型是CLR和编程语言提供的一种特殊机制,它支持另一种形式的代码重用.泛型通常用与集合以及作用于集合的方法一起使用,当然也可以单独使用. C#是一 ...
- .NET委托解析(异步委托)
上一篇我们了解到了,委托的基本感念,列举了几个委托的实例,并根据实例来反编译源码查看.NET 委托的内部实现,从浅入深的角度来详细的去解析委托的实质,本文将系上篇继续讨论异步委托的实现以及异步委托的源 ...
- C++等语言中整型int等的取值范围计算方式
举short为例说明 如果以最高位为符号位,二进制原码最大为0111111111111111=2的15次方减1=32767.最小为1111111111111111=-2的15次方减1=-32767此时 ...
- 关于 android百度地图 调用 地理位置 经纬度坐标,只调用一次的解决方法,通知栏不总是 搜索 GPS 。。。
上代码吧... //读取当前坐标 final LocationClient mLocationClient = new LocationClient(mActivity); mLocationClie ...
- 5月25日-js操作DOM遍历子节点
一.遍历节点 遍历子节点 children();//获取节点的所有直接子类 遍历同辈节点 next(); prev(); siblings();//所有同辈元素 *find(); 从后代元素中查找匹配 ...