系统启动一个新线程需要很高的成本,因为它涉及与操作系统交互。在这种情况下,使用线程池可以很好地提高性能,尤其是当程序中需要创建大量生存期很短暂的线程时。

线程池在系统启动时即创建大量空闲的线程,程序将一个Runnable对象或Callable对象传给线程池,线程池就会启动一个线程来run()或call()方法,当run()或call()方法执行结束后,该线程并不会死亡,而是再次返回线程池中成为空闲状态,等待执行下一个Runnable对象的run()或call()方法。除此之外,使用线程池可以有效地控制系统中并发线程的数量。

使用Executors类来创建线程池:

方法名
newCachedThreadPool() 创建一个具有缓存功能的线程池,系统根据需要创建线程,这些线程将会被缓存在线程池中
newFixedThreadPool(int nThreads) 创建一个可重用的、具有固定线程数的线程池
newSingleThreadExecutor() 创建一个只有单线程的线程池,它相当与调用newFixedThreadPool()方法时传入参数为1
newScheduledThreadPool(int corePoolSize) 创建具有指定线程数的线程池,它可以在指定延迟后执行线程任务。corePoolSize指池中所保存的线程数,即使线程是空闲的也被保存在线程池内。
newSingleThreadScheduledExecutor() 创建只有一个线程的线程池,它可以在指定延迟后执行线程任务
ExecutorService newWorkStealingPool(int parallelism) 创建持有足够的线程的线程池来支持给定的并行级别,该方法还会使用多个队列来减少竞争
ExecutorService newWorkStealingPool() 该方法是前一个方法的 简化版本,如果当前机器有4个CPU,则目标并行级别被设置为4,也就是相当于为前一个方法传入4作为参数。

使用线程池来执行线程任务的步骤:

1.调用Executors类的静态工厂方法创建一个ExecutorService对象,该对象代表一个线程池。

2.创建Runnable实现类或Callable实现类的实例,作为线程执行任务。

3.调用ExecutorService对象的submit()方法来提交Runnable实例或Callable实例。

4.当不想提交任何任务时,调用ExecutorService对象的shutdown()方法来关闭线程池。

使用线程池来执行指定Runnable对象所代表的的任务:

public class ThreadPoolTest {

public static void main(String[] args)throws Exception{

//创建一个具有固定线程数(6)的线程池

ExecutorService pool = Executors.newFixedThreadPool(6);

//使用Lambda表达式创建Runnable对象
Runnable target = () -> {

for(int i = 0; i < 100; i++){

System.out.println(Thread.currentThread().getName() + "的i值为:" + i);

}

};

//向线程池中提交两个线程

pool.submit(target);

pool.submit(target);
//关闭线程池

pool.shutdown();

}

}

ForkJoinPool类

ForkJoinPool类的作用是将一个任务拆分为多个“小任务”并行计算,再把多个“小任务”的结果合并成总的计算结果。ForkJoinPool类是ExecutorService的实现类,因此是一种特殊的线程池。

方法
ForkJoinPool(int parallelism) 创建一个包含parallelism个并行线程的ForkJoinPool
ForkJoinPool() 以runtime.availableprocessors(处理器数量)返回值为parallelism参数来创建ForkJoinPool。
ForkJoinPool commonPool() 该方法返回一个通用池,通用池的运行状态不回受shutdown()或shutdownNow()方法的影响。
int getCommonPoolParallelism() 返回通用池的并行级别

创建了ForkJoinPool实例之后,就可调用ForkJoinPool的submit(ForkJoinTask task)或invoke(ForkJoinTask task)方法来执行指定任务。其中ForkJoinTask代表一个可以并行、合并的任务。ForkJoinTask是一个抽象类,它还有两个抽象子类:RecursiveAction(没有返回值)和RecursiveTask(有返回值)。

没有返回值的“大任务”拆分:

//继承RecursiveAction来实现“可分解”的任务
class PrintTask extends RecursiveAction{

//每个“小任务”最多打印50个数

private static final int
THRESHOLD = 50;

private int start;

private int end;

//打印从start到end的任务

public PrintTask(int start, int end){

this.start = start;

this.end = end;

}

@Override

protected void compute() {

// 当end与start之间的差小于THRESHOLD时,开始打印

if(end - start < THRESHOLD){

for(int i = 0; i < end; i++){

System.out.println(Thread.currentThread().getName() + "的i值:" + i);

}

}else{

//当end与start之间的差大于THRESHOLD时,开始打印

//将大任务分解成两个“小任务”

int middle = (start + end) / 2;

PrintTask left = new PrintTask(start, middle);

PrintTask right = new PrintTask(middle, end);

//并执行两个“小任务”

left.fork();

right.fork();

}

}

}

public class ForkJoinPoolTest { public static void main(String[] args)throws Exception{

ForkJoinPool pool = new ForkJoinPool();

//提交可分解的PrintTask任务

pool.submit(new PrintTask(0, 300));

pool.awaitTermination(2, TimeUnit.SECONDS);

//关闭线程池

pool.shutdown();

}

}

有返回值的“大任务”:

//继承RecursiveTask来实现“可分解”的任务
class CalTask extends RecursiveTask<Integer>{
//每个“小任务”最多只累加20个数

private static final int THRESHOLD = 20;

private int arr[];

private int start;

private int end;

//累加从start到end数组

public CalTask(int[] arr, int start, int end){

this.arr = arr;

this.start = start;

this.end = end;

}

@Override

protected Integer compute() {

int sum = 0;

// 当end与start之间的差小于THRESHOLD时,开始进行实际累加

if(end - start < THRESHOLD){

for(int i = start; i < end; i++){

sum += arr[i];

}

return sum;

}else{

//当end与start之间的差大于THRESHOLD,既要累加的数超过20个时,将大任务分解成两个小任务

int middle = (start + end) / 2;
CalTask left = new CalTask(arr, start, middle);

CalTask right = new CalTask(arr, middle, end);

//并执行两个小任务
left.fork();

right.fork();

//把两个小任务累加的结果合并起来

return left.join() + right.join();

}

}



}

public class Sum {
public static void main(String[] args)throws Exception{

int[] arr = new int[100];
Random rand = new Random();

int total = 0;

//初始化100个数字元素

for(int i = 0, len = arr.length; i < len; i++){

int tmp = rand.nextInt(20);

//对数组元素赋值,并将数组元素的值添加到sum总和中

System.out.println(tmp);

total += (arr[i] = tmp);

}

System.out.println("初始化统计出的和:" + total);

//创建一个通用池

ForkJoinPool pool = ForkJoinPool.commonPool();

//提交可分解的CaltTask任务

Future<Integer> future = pool.submit(new CalTask(arr, 0, arr.length));

System.out.println("使用CalTask计算出的和:" + future.get());

//关闭线程

pool.shutdown();

}

}

Java线程的学习_线程池的更多相关文章

  1. java多线程面试题_线程并发面试题

    1.什么是线程?线程是操作系统能够进行运算调度的最小单位,它被包含在进程之中,是进程中的实际运作单位.程序员可以通过它进行多处理器编程,你可以使用多线程对运算密集型任务提速.比如,如果一个线程完成一个 ...

  2. Java并发编程学习:线程安全与锁优化

    本文参考<深入理解java虚拟机第二版> 一.什么是线程安全? 这里我借<Java Concurrency In Practice>里面的话:当多个线程访问一个对象,如果不考虑 ...

  3. java并发编程学习: 守护线程(Daemon Thread)

    在正式理解这个概念前,先把 守护线程 与 守护进程 这二个极其相似的说法区分开,守护进程通常是为了防止某些应用因各种意外原因退出,而在后台独立运行的系统服务或应用程序. 比如:我们开发了一个邮件发送程 ...

  4. java核心知识点学习----创建线程的第三种方式Callable和Future CompletionService

    前面已经指出通过实现Runnable时,Thread类的作用就是将run()方法包装成线程执行体,那么是否可以直接把任意方法都包装成线程执行体呢?Java目前不行,但其模仿者C#中是可以的. Call ...

  5. JAVA并发编程学习笔记------线程的三种创建方式

    创建线程一般有如下几个方式: 1. 通过继承Thread类来创建一个线程: /** * 步骤1:定义一个继承Thread类的子类 * 步骤2:构造子类的一个对象 * 步骤3:启动线程: * */ pu ...

  6. Java学习笔记 线程池使用及详解

    有点笨,参考了好几篇大佬们写的文章才整理出来的笔记.... 字面意思上解释,线程池就是装有线程的池,我们可以把要执行的多线程交给线程池来处理,和连接池的概念一样,通过维护一定数量的线程池来达到多个线程 ...

  7. Java并发包源码学习之线程池(一)ThreadPoolExecutor源码分析

    Java中使用线程池技术一般都是使用Executors这个工厂类,它提供了非常简单方法来创建各种类型的线程池: public static ExecutorService newFixedThread ...

  8. java学习之线程池的实现

    package com.gh.threadpoor; import java.util.concurrent.ExecutorService; import java.util.concurrent. ...

  9. java线程API学习 线程池ThreadPoolExecutor(转)

    线程池ThreadPoolExecutor继承自ExecutorService.是jdk1.5加入的新特性,将提交执行的任务在内部线程池中的可用线程中执行. 构造函数 ThreadPoolExecut ...

随机推荐

  1. 洛谷 P2353 背单词

    题目背景 小明对英语一窍不通,令老师十分头疼.于是期末考试前夕,小明被逼着开始背单词…… 题目描述 老师给了小明一篇长度为N的英语文章,然后让小明背M个单词.为了确保小明不会在背单词时睡着,老师会向他 ...

  2. Docker镜像的目录存储讲解

    我们成功安装完docker后,执行命令行sudo docker run hello-world, 如果是第一次执行,则会从远程拉取hello-world的镜像到本地,然后运行,显示hello worl ...

  3. HDU 4341 Gold miner (分组背包)

    先把线按照距离原点的距离排序,然后用叉积把在同一条直线上的点放在一起, 把在同一条线上的点中的前i个点当成一个点就转化成了分组背包. 写if(kas++) putchar('\n') 居然PE了,PE ...

  4. 搭建SSI开发框架原理

    Spring2.5.Struts2.Ibatis开发框架搭建(一) ssi, ibatis 一.框架下载 1.1   Struts2框架 Struts2框架发展于WebWork,现在捐献给了Apach ...

  5. IAP介绍

    iOS应用调置 wjforstudy分享了IAP的一些基本知识.在论坛的地址是:http://www.cocoachina.com/bbs/read.php?tid=92060  1.在开始IAP开发 ...

  6. 如何用纯 CSS 创作一个雷达扫描动画

    效果预览 在线演示 按下右侧的"点击预览"按钮可以在当前页面预览,点击链接可以全屏预览. https://codepen.io/comehope/pen/VdbGvr 可交互视频 ...

  7. (原)剑指offer变态跳台阶

    变态跳台阶 时间限制:1秒空间限制:32768K 题目描述 一只青蛙一次可以跳上1级台阶,也可以跳上2级……它也可以跳上n级.求该青蛙跳上一个n级的台阶总共有多少种跳法.   分析一下明天是个斐波那契 ...

  8. 排序算法C语言实现——快速排序的递归和非递归实现

    /*快排 -  递归实现nlogn*//*原理:    快速排序(Quicksort)是对冒泡排序的一种改进.    快速排序由C. A. R. Hoare在1962年提出.它的基本思想是:通过一趟排 ...

  9. PAT Basic 1030

    1030 完美数列 给定一个正整数数列,和正整数p,设这个数列中的最大值是M,最小值是m,如果M <= m * p,则称这个数列是完美数列. 现在给定参数p和一些正整数,请你从中选择尽可能多的数 ...

  10. 算法学习记录-图——应用之拓扑排序(Topological Sort)

    这一篇写有向无环图及其它的应用: 清楚概念: 有向无环图(DAG):一个无环的有向图.通俗的讲就是从一个点沿着有向边出发,无论怎么遍历都不会回到出发点上. 有向无环图是描述一项工程或者系统的进行过程的 ...