Java多线程设计模式之线程池模式
前序:
Thread-Per-Message Pattern,是一种对于每个命令或请求,都分配一个线程,由这个线程执行工作。它将“委托消息的一端”和“执行消息的一端”用两个不同的线程来实现。该线程模式主要包括三个部分:
1,Request参与者(委托人),也就是消息发送端或者命令请求端
2,Host参与者,接受消息的请求,负责为每个消息分配一个工作线程。
3,Worker参与者,具体执行Request参与者的任务的线程,由Host参与者来启动。
由于常规调用一个方法后,必须等待该方法完全执行完毕后才能继续执行下一步操作,而利用线程后,就不必等待具体任务执行完毕,就可以马上返回继续执行下一步操作。
背景:
由于在Thread-Per-Message Pattern中对于每一个请求都会生成启动一个线程,而线程的启动是很花费时间的工作,所以鉴于此,提出了Worker Thread,重复利用已经启动的线程。
线程池:
Worker Thread,也称为工人线程或背景线程,不过一般都称为线程池。该模式主要在于,事先启动一定数目的工作线程。当没有请求工作的时候,所有的工人线程都会等待新的请求过来,一旦有工作到达,就马上从线程池中唤醒某个线程来执行任务,执行完毕后继续在线程池中等待任务池的工作请求的到达。
任务池:主要是存储接受请求的集合,利用它可以缓冲接受到的请求,可以设置大小来表示同时能够接受最大请求数目。这个任务池主要是供线程池来访问。
线程池:这个是工作线程所在的集合,可以通过设置它的大小来提供并发处理的工作量。对于线程池的大小,可以事先生成一定数目的线程,根据实际情况来动态增加或者减少线程数目。线程池的大小不是越大越好,线程的切换也会耗时的。
存放池的数据结构,可以用数组也可以利用集合,在集合类中一般使用Vector,这个是线程安全的。
Worker Thread的所有参与者:
1,Client参与者,发送Request的参与者
2,Channel参与者,负责缓存Request的请求,初始化启动线程,分配工作线程
3,Worker参与者,具体执行Request的工作线程
4,Request参与者
注意:将在Worker线程内部等待任务池非空的方式称为正向等待。
将在Channel线程提供Worker线程来判断任务池非空的方式称为反向等待。
线程池实例1:
利用同步方法来实现,使用数组来作为任务池的存放数据结构。在Channel有缓存请求方法和处理请求方法,利用生成者与消费者模式来处理存储请求,利用反向等待来判断任务池的非空状态。
Channel参与者:
|
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
|
package whut.threadpool;//用到了生产者与消费者模式//生成线程池,接受客户端线程的请求,找到一个工作线程分配该客户端请求public class Channel private static final int MAX_REQUEST100;// //利用数组来存放请求,每次从数组末尾添加请求,从开头移除请求来处理 private final Request[]// private int tail;//下一次存放Request的位置 private int head;//下一次获取Request的位置 private int count;// private final WorkerThread[]// // public Channel(int threads) this.requestQueuenew Request[MAX_REQUEST]; this.head0; this.head0; this.count0; threadPoolnew WorkerThread[threads]; // for (int i0; threadPool[i]new WorkerThread("Worker-" +this); } } public void startWorkers() for (int i0; threadPool[i].start(); } } // public synchronized void putRequest(Request // while (count try { wait(); } catch (InterruptedException } requestQueue[tail] tail1) count++; notifyAll(); } // public synchronized Request while (count0) try { wait(); } catch (InterruptedException } Request head1) count--; notifyAll(); return request; }} |
客户端请求线程:
|
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
|
package whut.threadpool;import java.util.Random;//向Channel发送Request请求的public class ClientThread extends Thread{ private final Channel private static final Randomnew Random(); public ClientThread(String { super(name); this.channel=channel; } public void run() { try{ for(int i=0;true;i++) { Requestnew Request(getName(),i); channel.putRequest(request); Thread.sleep(random.nextInt(1000)); } }catch(InterruptedException { } }} |
工作线程:
|
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
|
package whut.threadpool;//具体工作线程public class WorkerThread extends Thread{ private final Channel public WorkerThread(String { super(name); this.channel=channel; } public void run() { while(true) { Request request.execute(); } }} |
线程池实例2:
工作线程:
利用同步块来处理,利用Vector来存储客户端请求。在Channel有缓存请求方法和处理请求方法,利用生成者与消费者模式来处理存储请求,利用正向等待来判断任务池的非空状态。
这种实例,可以借鉴到网络ServerSocket处理用户请求的模式中,有很好的扩展性与实用性。
利用Vector来存储,依旧是每次集合的最后一个位置添加请求,从开始位置移除请求来处理。
Channel参与者:
|
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
|
package whut.threadpool2;import java.util.Vector;/* * * * * * */public class Channel public final static int THREAD_COUNT=4; public static void main(String[] //定义两个集合,一个是存放客户端请求的,利用Vector, //一个是存储线程的,就是线程池中的线程数目 //Vector是线程安全的,它实现了Collection和List //Vector //它包含可以使用整数索引进行访问的组件。但Vector //以适应创建 //Collection中主要包括了list相关的集合以及set相关的集合,Queue相关的集合 //注意:Map不是Collection的子类,都是java.util.*下的同级包 Vectornew Vector(); //工作线程,初始分配一定限额的数目 WorkerThread[]new WorkerThread[THREAD_COUNT]; //初始化启动工作线程 for(int i=0;i<workers.length;i++) { workers[i]=new WorkerThread(pool); workers[i].start(); } //接受新的任务,并且将其存储在Vector中 Objectnew Object();//模拟的任务实体类 //此处省略具体工作 //在网络编程中,这里就是利用ServerSocket来利用ServerSocket.accept接受一个Socket从而唤醒线程 //当有具体的请求达到 synchronized(pool) { pool.add(pool.size(), pool.notifyAll();//通知所有在pool } //注意上面这步骤添加任务池请求,以及通知线程,都可以放在工作线程内部实现 //只需要定义该方法为static,在方法体用同步块,且共享的线程池也是static即可 //下面这步,可以有可以没有根据实际情况 //取消等待的线程 for(int i=0;i<workers.length;i++) { workers[i].interrupt(); } }} |
工作线程:
|
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
|
package whut.threadpool2;import java.util.List;public class WorkerThread extends Thread private List//任务请求池 private static int fileCompressed=0;//所有实例共享的 public WorkerThread(List { this.pool=pool; } //利用静态synchronized来作为整个synchronized类方法,仅能同时一个操作该类的这个方法 private static synchronized void incrementFilesCompressed() { fileCompressed++; } public void run() { //保证无限循环等待中 while(true) { //共享互斥来访问pool变量 synchronized(pool) { //利用多线程设计模式中的 //Guarded while(pool.isEmpty()) { try{ pool.wait();//进入pool的wait }catch(InterruptedException { } } //当线程被唤醒,需要重新获取pool的锁, //再次继续执行synchronized代码块中其余的工作 //当不为空的时候,继续再判断是否为空,如果不为空,则跳出循环 //必须先从任务池中移除一个任务来执行,统一用从末尾添加,从开始处移除 pool.remove(0);//获取任务池中的任务,并且要进行转换 } //下面是线程所要处理的具体工作 } }} |
Java多线程设计模式之线程池模式的更多相关文章
- Java多线程设计模式(4)线程池模式
前序: Thread-Per-Message Pattern,是一种对于每个命令或请求,都分配一个线程,由这个线程执行工作.它将“委托消息的一端”和“执行消息的一端”用两个不同的线程来实现.该线程模式 ...
- Java多线程系列--“JUC线程池”06之 Callable和Future
概要 本章介绍线程池中的Callable和Future.Callable 和 Future 简介示例和源码分析(基于JDK1.7.0_40) 转载请注明出处:http://www.cnblogs.co ...
- Java多线程系列--“JUC线程池”02之 线程池原理(一)
概要 在上一章"Java多线程系列--“JUC线程池”01之 线程池架构"中,我们了解了线程池的架构.线程池的实现类是ThreadPoolExecutor类.本章,我们通过分析Th ...
- Java多线程系列--“JUC线程池”03之 线程池原理(二)
概要 在前面一章"Java多线程系列--“JUC线程池”02之 线程池原理(一)"中介绍了线程池的数据结构,本章会通过分析线程池的源码,对线程池进行说明.内容包括:线程池示例参考代 ...
- Java多线程系列--“JUC线程池”04之 线程池原理(三)
转载请注明出处:http://www.cnblogs.com/skywang12345/p/3509960.html 本章介绍线程池的生命周期.在"Java多线程系列--“基础篇”01之 基 ...
- Java多线程系列--“JUC线程池”05之 线程池原理(四)
概要 本章介绍线程池的拒绝策略.内容包括:拒绝策略介绍拒绝策略对比和示例 转载请注明出处:http://www.cnblogs.com/skywang12345/p/3512947.html 拒绝策略 ...
- 转:java多线程CountDownLatch及线程池ThreadPoolExecutor/ExecutorService使用示例
java多线程CountDownLatch及线程池ThreadPoolExecutor/ExecutorService使用示例 1.CountDownLatch:一个同步工具类,它允许一个或多个线程一 ...
- Java多线程-新特性-线程池
Sun在Java5中,对Java线程的类库做了大量的扩展,其中线程池就是Java5的新特征之一,除了线程池之外,还有很多多线程相关的内容,为多线程的编程带来了极大便利.为了编写高效稳定可靠的多线程程序 ...
- Java多线程之细说线程池
前言 在认识线程池之前,我们需要使用线程就去创建一个线程,但是我们会发现有一个问题: 如果并发的线程数量很多,并且每个线程都是执行一个时间很短的任务就结束了,这样频繁创建线程就会大大降低系统的效率,因 ...
- Java多线程系列--“JUC线程池”01之 线程池架构
概要 前面分别介绍了"Java多线程基础"."JUC原子类"和"JUC锁".本章介绍JUC的最后一部分的内容——线程池.内容包括:线程池架构 ...
随机推荐
- nacos 注册服务正常,服务列表内没有服务
解决办法: 在nacos里 创建 注册的 命名空间 , 然后再看服务列表 就会有对应命名空间的数据了
- express项目的创建
前言 前端开发者若要进行后端开发,大多都会选择node.js,在node生态下是有大量框架的,其中最受新手喜爱的便是老牌的express.js,接下来我们就从零创建一个express项目. 安装nod ...
- 【YashanDB知识库】oracle dblink varchar类型查询报错记录
问题单:Oracle DBLINK查询崖山DB报错 oracle服务器上ODBC安装 unixodbc安装:yum -y install unixODBC mysql 配置安装对应版本的odbc: m ...
- 模N取余法实现大整数进制转换 ——C语言版(2-16进制均可)
思路如标题所说采用模N取余法,难点是这个除法过程如何实现. 个人推荐先到这篇博客学习一下,大佬的思路就是不一样:大数除法--超详细讲解 我所做的就是在上面博文代码的基础上增加了循环,用一个字符数组逆序 ...
- [C#基础1/21] C#概述
Notion原笔记 1. C# 简介 1.1 C# 定义 C# 在继承 C 和 C++ 强大功能的同时去掉了一些它们的复杂特性,使其成为 C 语言家族中的一种高效强大的编程语言 1.2 C# 用途 用 ...
- Argo CD初体验
什么是 Argo CD? Argo CD 是一个声明式的 GitOps 持续交付工具,用于 Kubernetes 集群.它通过持续监控 Git 仓库中的 Kubernetes 资源配置文件,将这些配置 ...
- Redis 入门 - 安装最全讲解(Windows、Linux、Docker)
经过上一章节的介绍,相信大家对Redis已经有了大致的认知,今天主要给大家详细讲解Redis在Windows.Linux.Docker下的安装过程. 01.Windows 下面给大家介绍三种在Wind ...
- 中文关键字检索分析-导出到csv或者excel-多文件或文件夹-使用python和asyncio和pandas的dataframe
1.02版本 把原来的tab一个个拼接成文件输出,改成pandas的dataframe 使用asyncio库来使用协程,但是测试下来速度好像是差不多的.可能速度太快了,没能很好的测出来差异. 原来的最 ...
- opencv equalizeHist
''' What are histograms? Histograms are collected counts of data organized into a set of predefined ...
- C# – Record, Class, Struct
前言 之前在 C# – 10.0 已经有介绍过 Record 了. 但之前还没怎么用到, 最近有用到了, 所以特别写多一篇. Class vs Struct 参考: C#详解struct和class的 ...