Master-Worker是常用的并行计算模式。它的核心思想是系统由两类进程协作工作:Master进程和Worker进程。Master负责接收和分配任务,Worker负责处理子任务。当各个Worker子进程处理完成后,会将结果返回给Master,由Master作归纳总结。其好处就是能将一个大任务分解成若干个小任务,并行执行,从而提高系统的吞吐量。处理过程如下图所示:

Master进程为主要进程,它维护一个Worker进程队列、子任务队列和子结果集。Worker进程队列中的Worker进程不停从任务队列中提取要处理的子任务,并将结果写入结果集。

根据上面的思想,我们来模拟一下这种经典设计模式的实现。

分析过程:

  1. 既然Worker是具体的执行任务,那么Worker一定要实现Runnable接口
  2. Matser作为接受和分配任务,得先有个容器来装载用户发出的请求,在不考虑阻塞的情况下我们选择ConcurrentLinkedQueue作为装载容器
  3. Worker对象需要能从Master接收任务,它也得有Master ConcurrentLinkedQueue容器的引用
  4. Master还得有个容器需要能够装载所有的Worker,可以使用HashMap<String,Thread>
  5. Worker处理完后需要将数据返回给Master,那么Master需要有个容器能够装载所有worker并发处理任务的结果集。此容器需要能够支持高并发,所以最好采用ConcurrentHashMap<String,Object>
  6. 同理由于Worker处理完成后将数据填充进Master的ConcurrentHashMap,那么它也得有一份ConcurrentHashMap的引用 

代码实现:

Task任务对象

  1. public class Task {
  2. private int id;
  3. private String name;
  4. private int price;
  5.  
  6. public int getId() {
  7. return id;
  8. }
  9.  
  10. public void setId(int id) {
  11. this.id = id;
  12. }
  13.  
  14. public String getName() {
  15. return name;
  16. }
  17.  
  18. public void setName(String name) {
  19. this.name = name;
  20. }
  21.  
  22. public int getPrice() {
  23. return price;
  24. }
  25.  
  26. public void setPrice(int price) {
  27. this.price = price;
  28. }
  29. }

Master对象:

  1. public class Master {
  2. //任务集合
  3. private ConcurrentLinkedQueue<Task> taskQueue = new ConcurrentLinkedQueue<>();
  4.  
  5. //所有的处理结果
  6. private ConcurrentHashMap<String,Object> resultMap = new ConcurrentHashMap<>();
  7.  
  8. //所有的Worker集合
  9. private HashMap<String,Thread> workerMap = Maps.newHashMap();
  10.  
  11. //构造方法,初始化Worker
  12. public Master(Worker worker,int workerCount){
  13. //每一个worker对象都需要有Master的引用,taskQueue用于任务的提取,resultMap用于任务的提交
  14. worker.setTaskQueue(this.taskQueue);
  15. worker.setResultMap(this.resultMap);
  16. for(int i = 0 ;i < workerCount; i++){
  17. //key表示worker的名字,value表示线程执行对象
  18. workerMap.put("worker"+i,new Thread(worker));
  19. }
  20. }
  21.  
  22. //用于提交任务
  23. public void submit(Task task){
  24. this.taskQueue.add(task);
  25. }
  26.  
  27. //执行方法,启动应用程序让所有的Worker工作
  28. public void execute(){
  29. for(Map.Entry<String,Thread> me : workerMap.entrySet()){
  30. me.getValue().start();
  31. }
  32. }
  33.  
  34. //判断所有的线程是否都完成任务
  35. public boolean isComplete() {
  36. for(Map.Entry<String,Thread> me : workerMap.entrySet()){
  37. if(me.getValue().getState() != Thread.State.TERMINATED){
  38. return false;
  39. }
  40. }
  41. return true;
  42. }
  43.  
  44. //总结归纳
  45. public int getResult(){
  46. int ret = 0;
  47. for (Map.Entry<String, Object> entry : resultMap.entrySet()) {
  48. ret+=(Integer) entry.getValue();
  49. }
  50. return ret;
  51. }
  52. }

Worker对象:

  1. public class Worker implements Runnable{
  2. private ConcurrentLinkedQueue<Task> taskQueue;
  3. private ConcurrentHashMap<String, Object> resultMap;
  4.  
  5. public void setTaskQueue(ConcurrentLinkedQueue<Task> taskQueue) {
  6. this.taskQueue = taskQueue;
  7. }
  8.  
  9. public void setResultMap(ConcurrentHashMap<String, Object> resultMap) {
  10. this.resultMap = resultMap;
  11. }
  12.  
  13. @Override
  14. public void run() {
  15. while(true){
  16. Task executeTask = this.taskQueue.poll();
  17. if(executeTask == null) break;
  18. //真正的任务处理
  19. Object result = handle(executeTask);
  20. this.resultMap.put(executeTask.getName(),result);
  21. }
  22. }
  23.  
  24. //核心处理逻辑,可以抽离出来由具体子类实现
  25. private Object handle(Task executeTask) {
  26. Object result = null;
  27. try {
  28. //表示处理任务的耗时....
  29. Thread.sleep(500);
  30. result = executeTask.getPrice();
  31. } catch (InterruptedException e) {
  32. e.printStackTrace();
  33. }
  34. return result;
  35. }
  36. }

客户端调用:

  1. public class Main {
  2.  
  3. public static void main(String[] args) {
  4. //实际开发中多少个线程最好写成Runtime.getRuntime().availableProcessors()
  5. Master master = new Master(new Worker(), 10);
  6. Random random = new Random();
  7. for(int i = 0 ;i <= 100 ;i++){
  8. Task task = new Task();
  9. task.setId(i);
  10. task.setName("任务"+i);
  11. task.setPrice(random.nextInt(1000));
  12. master.submit(task);
  13. }
  14. master.execute();
  15. long start = System.currentTimeMillis();
  16. while(true){
  17. if(master.isComplete()){
  18. long end = System.currentTimeMillis() - start;
  19. int ret = master.getResult();
  20. System.out.println("计算结果:"+ret+",执行耗时:"+end);
  21. break;
  22. }
  23. }
  24. }
  25. }

在Worker对象中的核心处理业务逻辑handle()方法最好抽象成公共方法,具体实现由子类覆写。

多线程设计模式 : Master-Worker模式的更多相关文章

  1. 多线程设计模式——Read-Write Lock模式和Future模式分析

    目录 多线程程序评价标准 任何模式都有一个相同的"中心思想" Read-Write Lock 模式 RW-Lock模式特点 冲突总结 手搓RW Lock模式代码 类图 Data类 ...

  2. 14.多线程设计模式 - Master-Worker模式

    多线程设计模式 - Master-Worker模式 并发设计模式属于设计优化的一部分,它对于一些常用的多线程结构的总结和抽象.与串行相比并行程序结构通常较为复杂,因此合理的使用并行模式在多线程并发中更 ...

  3. Java多线程设计模式(4)线程池模式

    前序: Thread-Per-Message Pattern,是一种对于每个命令或请求,都分配一个线程,由这个线程执行工作.它将“委托消息的一端”和“执行消息的一端”用两个不同的线程来实现.该线程模式 ...

  4. 多线程:多线程设计模式(三):Master-Worker模式

    Master-Worker模式是常用的并行模式之一,它的核心思想是,系统有两个进程协作工作:Master进程,负责接收和分配任务:Worker进程,负责处理子任务.当Worker进程将子任务处理完成后 ...

  5. 多线程设计模式(三):Master-Worker模式

    Master-Worker模式是常用的并行模式之一,它的核心思想是,系统有两个进程协作工作:Master进程,负责接收和分配任务:Worker进程,负责处理子任务.当Worker进程将子任务处理完成后 ...

  6. 多线程设计模式 - Future模式

    Future模式是多线程开发中非常常见的一种设计模式,它的核心思想是异步调用.这类似我们日常生活中的在线购物流程,带在购物网看着一件商品时可以提交表单,当订单完成后就可以在家里等待商品送货上门.或者说 ...

  7. 13.多线程设计模式 - Future模式

    多线程设计模式 - Future模式 并发设计模式属于设计优化的一部分,它对于一些常用的多线程结构的总结和抽象.与串行相比并行程序结构通常较为复杂,因此合理的使用并行模式在多线程并发中更具有意义. 1 ...

  8. Master和worker模式

    让和hadoop的设计思想是一样的,Master负责分配任务和获取任务的结果,worker是真正处理业务逻辑的. 使用ConcurrentLikedQueue去承载所有的任务,因为会有多个worker ...

  9. java多线程设计模式

    详见:http://blog.yemou.net/article/query/info/tytfjhfascvhzxcyt220 java多线程设计模式 java语言已经内置了多线程支持,所有实现Ru ...

随机推荐

  1. 02.VUE学习二之数据绑定

    <!DOCTYPE html> <html> <head> <meta charset="utf-8"> <meta http ...

  2. HDU1272小希的迷宫

    小希的迷宫 上次Gardon的迷宫城堡小希玩了很久(见Problem B),现在她也想设计一个迷宫让Gardon来走.但是她设计迷宫的思路不一样,首先她认为所有的通道都应该是双向连通的,就是说如果有一 ...

  3. ajax提交表单,支持文件上传

    当我们提交表单但是又不想要刷新页面的时候就可以考虑使用ajax来实现提交功能,但是这有个局限就是当有文件上传的时候是行不通的,下面借助于jquery.form可以很方便满足我们的需求.   1.表单写 ...

  4. 【Valid Sudoku】cpp

    题目: Determine if a Sudoku is valid, according to: Sudoku Puzzles - The Rules. The Sudoku board could ...

  5. 【Median of Two Sorted Arrays】cpp

    题目: There are two sorted arrays A and B of size m and n respectively. Find the median of the two sor ...

  6. python 学习分享-常用模块篇

    模块 就是前人给你造的轮子,你开车就好!!! 常用模块有: time模块 random模块 os模块 sys模块 shutil模块 json  &  picle模块 shelve模块 xml处 ...

  7. http的一些知识

    TCP/IP协议分层 应用层 TFP DNS DNS域名解析的过程 在浏览器DNS缓存中搜索 读取系统的hosts文件,查找其中是否有对应的ip 向本地配置的首选DNS服务器发起域名解析请求 HTTP ...

  8. 习题:玛丽卡(SPFA)

    玛丽卡(wikioi1021) [题目描述 ]麦克找了个新女朋友,玛丽卡对他非常恼火并伺机报复.因为她和他们不住在同一个城市,因此她开始准备她的长途旅行.在这个国家中每两个城市之间最多只有一条路相通, ...

  9. 【距离GDOI:141天】 滚入数位DP的坑

    作为博客园的第一篇...我都不知道要写什么了 ... 其实今天很没状态,就当吐槽吧... 嗯,被黄神带去写treap+可持久化线段树,然后在可持久化的删除上面跪了两天,真的是一跪不起.我已经连续多久没 ...

  10. [POI2005][luogu3462] SZA-Template [fail树]

    题面 传送门 思路 首先,我们观察一下这个要求的"模板串",发现它有如下性质: 1.一个模板串$A$是要求的文本串$B$的公共前后缀 2.如果一个模板串$A$有另一个模板串$B$( ...