java forkjoin MySQL_fork/join使用示例

本文介绍了Fork/Join框架的基本原理与应用,包括如何利用该框架实现分治法解决问题,核心类的作用及其实现细节,并通过三个示例展示了同步与异步任务的执行过程。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

fork/join框架是用多线程的方式实现分治法来解决问题。fork指的是将问题不断地缩小规模,join是指根据子问题的计算结果,得出更高层次的结果。

fork/join框架的使用有一定的约束条件:

1. 除了fork()  和  join()方法外,线程不得使用其他的同步工具。线程最好也不要sleep()

2. 线程不得进行I/O操作

3. 线程不得抛出checked exception

此框架有几个核心类:ForkJoinPool是实现了工作窃取算法的线程池。ForkJoinTask是任务类,他有2个子类:RecursiveAction无返回值,RecursiveTask有返回值,在定义自己的任务时,一般都是从这2类中挑一个,通过继承的方式定义自己的新类。由于ForkJoinTask类实现了Serializable接口,因此,定义自己的任务类时,应该定义serialVersionUID属性。

在编写任务时,推荐的写法是这样的:

If (problem size > defaultsize){

task s=divide(task);

execute(tasks);

}else{

resolve problem using another algorithm;

}

ForkJoinPool实现了工作窃取算法(work-stealing),线程会主动寻找新创建的任务去执行,从而保证较高的线程利用率。它使用守护线程(deamon)来执行任务,因此无需对他显示的调用shutdown()来关闭。一般情况下,一个程序只需要唯一的一个ForkJoinPool,因此应该按如下方式创建它:

static final ForkJoinPool mainPool = new ForkJoinPool(); //线程的数目等于CPU的核心数

下面给出一个非常简单的例子,功能是将一个数组中每一个元素的值加1。具体实现为:将大数组不断分解为更短小的子数组,当子数组长度不超过10的时候,对其中所有元素进行加1操作。

packageforkjoin;importjava.util.concurrent.ForkJoinPool;importjava.util.concurrent.RecursiveAction;public classTest {public final static ForkJoinPool mainPool = newForkJoinPool();public static voidmain(String[] args) {int n = 26;int[] a = new int[n];

System.out.println("before:");for (int i = 0; i < n; i++) {

a[i]=i;

System.out.print(a[i]+ " ");

}

SubTask task= new SubTask(a, 0, n);

mainPool.invoke(task);

System.out.println();

System.out.println("after:");for (int i = 0; i < n; i++) {

System.out.print(a[i]+ " ");

}

}

}class SubTask extendsRecursiveAction {private static final long serialVersionUID = 1L;private int[] a;private intbeg;private intend;public SubTask(int[] a, int beg, intend) {super();this.a =a;this.beg =beg;this.end =end;

}

@Overrideprotected voidcompute() {if (end - beg > 10) {int mid = (beg + end) / 2;

SubTask t1= newSubTask(a, beg, mid);

SubTask t2= newSubTask(a, mid, end);

invokeAll(t1, t2);

}else{for (int i = beg; i < end; i++) {

a[i]= a[i] + 1;

}

}

}

}

结果:

before:

0 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

after:

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

例子2,任务拥有返回值。随机生成一个数组,每个元素均是0-999之间的整数,统计该数组中每个数字出现1的次数的和。

实现方法,将该数组不断的分成更小的数组,直到每个子数组的长度为1,即只包含一个元素。此时,统计该元素中包含1的个数。最后汇总,得到数组中每个数字共包含了多少个1。

packageforkjoin.demo2;importjava.util.Random;importjava.util.concurrent.ExecutionException;importjava.util.concurrent.ForkJoinPool;importjava.util.concurrent.RecursiveTask;public classTest {public final static ForkJoinPool mainPool = newForkJoinPool();public static voidmain(String[] args) {int n = 26;int[] a = new int[n];

Random rand= newRandom();

System.out.println("before:");for (int i = 0; i < n; i++) {

a[i]= rand.nextInt(1000);

System.out.print(a[i]+ " ");

}

SubTask task= new SubTask(a, 0, n);int count =mainPool.invoke(task);

System.out.println();

System.out.println("after:");for (int i = 0; i < n; i++) {

System.out.print(a[i]+ " ");

}

System.out.println("\n数组中共出现了" + count + "个1");

}

}class SubTask extends RecursiveTask{private static final long serialVersionUID = 1L;private int[] a;private intbeg;private intend;public SubTask(int[] a, int beg, intend) {super();this.a =a;this.beg =beg;this.end =end;

}

@OverrideprotectedInteger compute() {int result = 0;if (end - beg > 1) {int mid = (beg + end) / 2;

SubTask t1= newSubTask(a, beg, mid);

SubTask t2= newSubTask(a, mid, end);

invokeAll(t1, t2);try{

result= t1.get() +t2.get();

}catch (InterruptedException |ExecutionException e) {

e.printStackTrace();

}

}else{

result=count(a[beg]);

}returnresult;

}//统计一个整数中出现了几个1

private int count(intn) {int result = 0;while (n > 0) {if (n % 10 == 1) {

result++;

}

n= n / 10;

}returnresult;

}

}

结果:

before:466 581 913 818 611 871 10 748 903 797 830 426 887 198 416 945 592 409 993 408 368 663 117 120 802 510after:466 581 913 818 611 871 10 748 903 797 830 426 887 198 416 945 592 409 993 408 368 663 117 120 802 510数组中共出现了13个1

例子3,异步执行任务。前面两个例子都是同步执行任务,当启动任务后,主线程陷入了阻塞状态,直到任务执行完毕。若创建新任务后,希望当前线程能继续执行而非陷入阻塞,则需要异步执行。ForkJoinPool线程池提供了execute()方法来异步启动任务,而作为任务本身,可以调用fork()方法异步启动新的子任务,并调用子任务的join()方法来取得计算结果。需要注意的是,异步使用ForkJoin框架,无法使用“工作窃取”算法来提高线程的利用率,针对每个子任务,系统都会启动一个新的线程。

本例的功能是查找硬盘上某一类型的文件。给定文件扩展名后,将硬盘上所有该类型的文件名打印显示出来。作为主程序,启动任务后,继续显示任务的执行进度,每3秒钟打印显示一个黑点,表示任务在继续。最后,当所有线程都结束了,打印显示结果。

packageforkjoin.demo3;importjava.io.IOException;importjava.nio.file.DirectoryStream;importjava.nio.file.FileSystems;importjava.nio.file.Files;importjava.nio.file.Path;importjava.nio.file.Paths;importjava.util.ArrayList;importjava.util.List;importjava.util.concurrent.ForkJoinPool;importjava.util.concurrent.RecursiveTask;importjava.util.concurrent.TimeUnit;public classThreadLocalTest {public static void main(String[] args) throwsException {

Path p= Paths.get("D:/");

List roots = (List) FileSystems.getDefault().getRootDirectories();

List result = new ArrayList<>();

List tasks = new ArrayList<>();

ForkJoinPool pool= newForkJoinPool();for(Path root : roots) {

MyTask t= new MyTask(root, "pdf");

pool.execute(t);

tasks.add(t);

}

System.out.print("正在处理中");while (isAllDone(tasks) == false) {

System.out.print(". ");

TimeUnit.SECONDS.sleep(3);

}for(MyTask t : tasks) {

result.addAll(t.get());

}for(Path pp : result) {

System.out.println(pp);

}

}private static boolean isAllDone(Listtasks) {boolean result = true;for(MyTask t : tasks) {if (t.isDone() == false) {

result= false;break;

}

}returnresult;

}

}class MyTask extends RecursiveTask>{private static final long serialVersionUID = 1L;privatePath path;privateString fileExtention;publicMyTask(Path path, String fileExtention) {super();this.path =path;this.fileExtention =fileExtention;

}

@Overrideprotected Listcompute() {

List result = new ArrayList<>();try{

DirectoryStream paths =Files.newDirectoryStream(path);

List subTasks = new ArrayList<>();for(Path p : paths) {if(Files.isDirectory(p)) {

MyTask t= newMyTask(p, fileExtention);

t.fork();

subTasks.add(t);

}else if(Files.isRegularFile(p)) {if (p.toString().toLowerCase().endsWith("." +fileExtention)) {

result.add(p);

}

}

}for(MyTask t : subTasks) {

result.addAll(t.join());

}

}catch(IOException e) {

}returnresult;

}

}

结果:

正在处理中. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . .

### 多线程中 `fork...join`、`fork...join_any` 和 `fork...join_none` 的用法差异 在 SystemVerilog 中,`fork...join`、`fork...join_any` 和 `fork...join_none` 是用于实现多线程并发的结构。它们的主要区别在于如何控制子线程的执行和父线程的等待行为。 #### 1. `fork...join` `fork...join` 是最基本的并发结构。在这种结构中,父线程会等待所有子线程完成执行后才会继续运行[^1]。这意味着,父线程会被阻塞,直到所有的子线程都执行完毕。 示例代码: ```systemverilog initial begin fork #10 $display("Task A"); #20 $display("Task B"); join $display("All tasks completed"); end ``` 在这个例子中,父线程会等待 `Task A` 和 `Task B` 都完成后才输出 `"All tasks completed"`[^2]。 #### 2. `fork...join_any` `fork...join_any` 结构允许父线程在任意一个子线程完成后立即继续执行,而不需要等待所有子线程完成[^3]。其余未完成的子线程将继续执行,但父线程不再等待它们。 示例代码: ```systemverilog initial begin fork #10 $display("Task A"); #20 $display("Task B"); join_any $display("At least one task completed"); end ``` 在这个例子中,当 `Task A` 或 `Task B` 中的任何一个完成时,父线程就会输出 `"At least one task completed"` 并继续执行[^4]。 #### 3. `fork...join_none` `fork...join_none` 结构启动所有子线程后,父线程会立即继续执行,而不会等待任何子线程完成。这种结构通常用于异步操作,或者需要手动控制子线程的终止。 示例代码: ```systemverilog initial begin fork #10 $display("Task A"); #20 $display("Task B"); join_none $display("Tasks started, parent continues"); end ``` 在这个例子中,父线程会在启动 `Task A` 和 `Task B` 后立即输出 `"Tasks started, parent continues"` 并继续执行[^4]。 #### 总结 - `fork...join`:父线程等待所有子线程完成。 - `fork...join_any`:父线程在任意一个子线程完成后继续执行。 - `fork...join_none`:父线程启动子线程后立即继续执行,不等待任何子线程。 ### 注意事项 在使用 `fork...join_none` 时,需要注意子线程可能会在父线程结束之前被意外终止。因此,通常需要结合 `wait fork` 或其他机制来确保子线程的正确执行[^1]。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值