百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 技术分类 > 正文

Java并发编程(20):fork/join

ztj100 2024-12-07 18:46 19 浏览 0 评论

Fork/Join 是 Java 并发框架中的一部分,主要用于 分治 (divide and conquer) 的任务处理模式,尤其适合 大规模并行计算递归任务。Fork/Join 框架是从 Java 7 开始引入的,旨在最大限度地利用多核处理器的并行计算能力,以提高计算效率。

工作原理

Fork/Join 框架的核心思想是将一个大任务分解成多个小任务,分别执行,然后将它们的结果汇总。它包含两个主要步骤:

  1. Fork:将任务分解成更小的子任务,直到子任务足够简单,适合直接执行。
  2. Join:当子任务完成时,将它们的结果合并。

关键类

Fork/Join 框架主要由以下两个类组成:

  1. ForkJoinPool:类似于线程池,但专门用于执行分解任务的线程池。它管理并调度 ForkJoinTask。
  2. ForkJoinTask:是一个抽象类,代表一个可以并行执行并合并结果的任务。它有两个重要的子类: RecursiveTask:有返回值的任务。 RecursiveAction:没有返回值的任务。

Fork/Join 框架的特点

  • 工作窃取算法:ForkJoinPool 使用“工作窃取算法”来提高 CPU 使用率。当一个线程完成自己的任务时,它可以窃取其他线程的任务来执行,减少等待时间,提高效率。
  • 自动分治:框架会递归地分解任务,直到任务变得足够简单,然后并行执行这些小任务。
  • 线程复用:避免了频繁的线程创建和销毁,提升资源利用率。

示例:使用 Fork/Join 计算斐波那契数列

以下示例展示了如何使用 Fork/Join 框架来计算斐波那契数列:

import java.util.concurrent.RecursiveTask;
import java.util.concurrent.ForkJoinPool;

public class FibonacciTask extends RecursiveTask<Integer> {
    private final int n;

    public FibonacciTask(int n) {
        this.n = n;
    }

    @Override
    protected Integer compute() {
        // 基础情形:当 n 小于等于 1 时直接返回结果
        if (n <= 1) {
            return n;
        }

        // 拆分成两个子任务
        FibonacciTask fib1 = new FibonacciTask(n - 1);
        FibonacciTask fib2 = new FibonacciTask(n - 2);

        // 使用 fork 分裂出子任务
        fib1.fork(); // 异步执行 fib1

        // 合并任务:fib2 和 fib1 结果
        return fib2.compute() + fib1.join();
    }

    public static void main(String[] args) {
        // 创建 ForkJoinPool
        ForkJoinPool pool = new ForkJoinPool();

        // 创建 FibonacciTask 任务计算第 10 个斐波那契数
        FibonacciTask task = new FibonacciTask(10);

        // 执行任务并获取结果
        int result = pool.invoke(task);
        System.out.println("Fibonacci(10) = " + result);

        pool.shutdown();
    }
}

代码说明

  • RecursiveTask:FibonacciTask 继承了 RecursiveTask<Integer>,表示这是一个有返回值的任务。
  • compute():重写 compute() 方法,如果 n 足够小直接返回结果,否则分解成子任务 fib1 和 fib2。
  • fork() 和 join():fib1 使用 fork() 异步执行,而 fib2 使用 compute() 直接执行,fib1 的结果使用 join() 方法合并。
  • ForkJoinPool:创建了一个 ForkJoinPool 来管理和执行任务。

使用场景

Fork/Join 框架适用于以下类型的任务:

  1. 大规模数据处理:例如,排序、大规模数据处理等。
  2. 递归算法:适合用递归分治法处理的场景,如归并排序、快速排序、斐波那契数列等。
  3. 并行计算:多核 CPU 下的密集计算任务。

注意事项

  1. 任务分割的粒度:子任务的粒度不宜过细,否则会增加线程间通信和调度的开销。可以根据计算量设置一个阈值。
  2. 性能监控:过度分解任务可能导致性能下降,最好在调试时对计算任务的分解层级和执行时间进行监控。
  3. 递归深度控制:过深的递归可能会导致栈溢出,需确保任务分解不会过于深入。

相关推荐

离谱!写了5年Vue,还不会自动化测试?

前言大家好,我是倔强青铜三。是一名热情的软件工程师,我热衷于分享和传播IT技术,致力于通过我的知识和技能推动技术交流与创新,欢迎关注我,微信公众号:倔强青铜三。Playwright是一个功能强大的端到...

package.json 与 package-lock.json 的关系

模块化开发在前端越来越流行,使用node和npm可以很方便的下载管理项目所需的依赖模块。package.json用来描述项目及项目所依赖的模块信息。那package-lock.json和...

Github 标星35k 的 SpringBoot整合acvtiviti开源分享,看完献上膝盖

前言activiti是目前比较流行的工作流框架,但是activiti学起来还是费劲,还是有点难度的,如何整合在线编辑器,如何和业务表单绑定,如何和系统权限绑定,这些问题都是要考虑到的,不是说纯粹的把a...

Vue3 + TypeScript 前端研发模板仓库

我们把这个Vue3+TypeScript前端研发模板仓库的初始化脚本一次性补全到可直接运行的状态,包括:完整的目录结构所有配置文件研发规范文档示例功能模块(ExampleFeature)...

Vue 2迁移Vue 3:从响应式到性能优化

小伙伴们注意啦!Vue2已经在2023年底正式停止维护,再不升级就要面临安全漏洞没人管的风险啦!而且Vue3带来的性能提升可不是一点点——渲染速度快40%,内存占用少一半,更新速度直接翻倍!还在...

VUE学习笔记:声明式渲染详解,对比WEB与VUE

声明式渲染是指使用简洁的模板语法,声明式的方式将数据渲染进DOM系统。声明式是相对于编程式而言,声明式是面向对象的,告诉框架做什么,具体操作由框架完成。编程式是面向过程思想,需要手动编写代码完成具...

苏州web前端培训班, 苏州哪里有web前端工程师培训

前端+HTML5德学习内容:第一阶段:前端页面重构:PC端网站布局、HTML5+CSS3基础项目、WebAPP页面布局;第二阶段:高级程序设计:原生交互功能开发、面向对象开发与ES5/ES6、工具库...

跟我一起开发微信小程序——扩展组件的代码提示补全

用户自定义代码块步骤:1.HBuilderX中工具栏:工具-代码块设置-vue代码块2.通过“1”步骤打开设置文件...

JimuReport 积木报表 v1.9.3发布,免费可视化报表

项目介绍积木报表JimuReport,是一款免费的数据可视化报表,含报表、大屏和仪表盘,像搭建积木一样完全在线设计!功能涵盖:数据报表、打印设计、图表报表、门户设计、大屏设计等!...

软开企服开源的无忧企业文档(V2.1.3)产品说明书

目录1....

一款面向 AI 的下一代富文本编辑器,已开源

简介AiEditor是一个面向AI的下一代富文本编辑器。开箱即用、支持所有前端框架、支持Markdown书写模式什么是AiEditor?AiEditor是一个面向AI的下一代富文本编辑...

玩转Markdown(2)——抽象语法树的提取与操纵

上一篇玩转Markdown——数据的分离存储与组件的原生渲染发布,转眼已经鸽了大半年了。最近在操纵mdast生成md文件的时候,心血来潮,把玩转Markdown(2)给补上了。...

DeepseekR1+ollama+dify1.0.0搭建企业/个人知识库(入门避坑版)

找了网上的视频和相关文档看了之后,可能由于版本不对或文档格式不对,很容易走弯路,看完这一章,可以让你少踩三天的坑。步骤和注意事项我一一列出来:1,前提条件是在你的电脑上已配置好ollama,dify1...

升级JDK17的理由,核心是降低GC时间

升级前后对比升级方法...

一个vsCode格式化插件_vscode格式化插件缩进量

ESlint...

取消回复欢迎 发表评论: