jdk1.7推出的Fork/Join提高业务代码处理性能

jdk1.7之后推出了Fork/Join框架,其原理个人理解为:递归多线程并发处理业务代码,以下为我模拟我们公司业务代码做的一个案例,性能可提升75%:

**下边这个类是模拟现有业务代码写的**
package NoForkJoin;

import forkJoinTest.Row;
import forkJoinTest.Student; import java.util.ArrayList;
import java.util.List; /**
* @author liu_l
* @Title: ServiceImp
* @ProjectName workspace-idea
* @Description: TODO
* @date 2018/6/240:32
*/
public class ServiceImp { public static void main(String[] args) throws InterruptedException {
long s0 = System.currentTimeMillis();
//造业务数据
List<Student> list = new ArrayList<Student>();
for (int i = 0; i < 10000; i++) {
Student student = new Student();
student.setName("test1" + i);
student.setSax("man");
student.setTall((double) i);
list.add(student);
}
//开始业务数据处理
List<Row> rows = new ArrayList<Row>();
for (int i = 0; i < 10000; i++) {
Student student = list.get(i);
//模拟一条业务数据处理需耗时1毫秒
Thread.sleep(1);
Row row = new Row();
row.put("name", student.getName());
row.put("sax", student.getSax());
row.put("tall", student.getTall());
rows.add(row);
}
System.out.println("共处理业务对象:" + rows.size() + "个");
System.out.println("共耗时:" + (System.currentTimeMillis() - s0) + "毫秒");
}
}

运行结果如图: 
 
下面为采用fork/join框架来实现此功能:

student类:模拟业务对像
        package forkJoinTest;

        import java.io.Serializable;

        /**
* @author liu_l
* @Title: Student
* @ProjectName workspace-idea
* @Description: TODO
* @date 2018/6/2323:03
*/
public class Student implements Serializable { private String name; private Double tall; private String sax; public String getName() {
return name;
} public void setName(String name) {
this.name = name;
} public Double getTall() {
return tall;
} public void setTall(Double tall) {
this.tall = tall;
} public String getSax() {
return sax;
} public void setSax(String sax) {
this.sax = sax;
}
}
rows继承与HashMap,将业务对象组装为map格式:
        package forkJoinTest;

        import java.util.ArrayList;
import java.util.concurrent.ConcurrentHashMap; /**
* @author liu_l
* @Title: Row
* @ProjectName workspace-idea
* @Description: TODO
* @date 2018/6/2323:06
*/
public class Row extends ConcurrentHashMap{ public Student Student;
}
**重点:Fork/Join框架处理业务代码:**
        package forkJoinTest;

        import java.util.ArrayList;
import java.util.List;
import java.util.concurrent.RecursiveTask; /**
* @author liu_l
* @Title: ForkJoin
* @ProjectName workspace-idea
* @Description: TODO
* @date 2018/6/2323:09
*/
public class ForkJoinCode extends RecursiveTask<List<Row>>{ protected static int THREAD_HOLD = 50;
protected int start;//开始任务序号
protected int end;//结束任务序号
protected List<Student> datas; /**
* @Description: TODO
* @param:
* @author liu-lei
* @date 2018/6/23 23:19
*/
public static ForkJoinCode getInstance(int start, int end, List<Student> datas){
ForkJoinCode forkJoinCode = new ForkJoinCode();
forkJoinCode.start = start;
forkJoinCode.end = end;
forkJoinCode.datas = datas;
return forkJoinCode;
} @Override
protected List<Row> compute() {
List<Row> rows = new ArrayList<Row>();
boolean canCompute = (end - start) <= THREAD_HOLD;
if(canCompute){
for(int i = start; i <= end; i++){
tranfromT2Row(rows, i);
}
}else{
int middle = (start + end)/2;
ForkJoinCode leftForkJoin = ForkJoinCode.getInstance(start, middle, datas);
ForkJoinCode rightForkJoin = ForkJoinCode.getInstance(middle+1, end, datas);
leftForkJoin.fork();
rightForkJoin.fork();
List<Row> lResult = leftForkJoin.join();
List<Row> rResult = rightForkJoin.join();
rows.addAll(lResult);
rows.addAll(rResult);
}
return rows;
} /**
* @Description: 业务代码处理
* @param:
* @author liu-lei
* @date 2018/6/24 0:33
*/
public void tranfromT2Row(List<Row> rows, int i){
Student student = datas.get(i);
Row row = new Row();
row.put("name", student.getName());
row.put("sax", student.getSax());
row.put("tall", student.getTall());
try {
//模拟业务数据处理需耗时5毫秒
Thread.sleep(3);
} catch (InterruptedException e) {
e.printStackTrace();
}
rows.add(row);
};
Service方法进行调用:
        package forkJoinTest;

        import java.util.ArrayList;
import java.util.List;
import java.util.concurrent.ExecutionException;
import java.util.concurrent.ForkJoinPool;
import java.util.concurrent.Future; /**
* @author liu_l
* @Title: ServiceImp
* @ProjectName workspace-idea
* @Description: TODO
* @date 2018/6/240:12
*/
public class ServiceImp {
public static void main(String[] args) throws ExecutionException, InterruptedException {
long s0 = System.currentTimeMillis();
//造业务数据
List<Student> list = new ArrayList<Student>();
for(int i = 0; i < 10000; i ++){
Student student = new Student();
student.setName("test1" + i);
student.setSax("man");
student.setTall((double)i);
list.add(student);
}
//开始业务数据处理
ForkJoinPool pool = new ForkJoinPool();
ForkJoinCode studentForkJoinCode = ForkJoinCode.getInstance(0, list.size()-1, list);
Future<List<Row>> result = pool.submit(studentForkJoinCode);
System.out.println("共处理业务对象:" + result.get().size() + "个");
showPoolStates(pool);
System.out.println("共耗时:" + (System.currentTimeMillis() - s0) + "毫秒");
} /**
* @Description: 监控Fork/Join池相关方法
* @param:
* @author liu-lei
* @date 2018/6/24 0:43
*/
private static void showPoolStates(ForkJoinPool pool){
System.out.println("*******************");
System.out.println("线程池的worker线程数量:" + pool.getPoolSize());
System.out.println("当前执行任务的线程数量:" + pool.getActiveThreadCount());
System.out.println("没有被阻塞正在工作的线程:" + pool.getRunningThreadCount());
System.out.println("已经提交给池还没有开始执行的任务数:" + pool.getQueuedSubmissionCount());
System.out.println("已经提交给池开始执行的任务数:" + pool.getQueuedTaskCount());
System.out.println("线程偷取任务数:" + pool.getStealCount()); }
}

测试结果如下: 

讲个结果对比性能提升了63%: 

jdk1.7推出的Fork/Join提高业务代码处理性能的更多相关文章

  1. 如何使用 Set 来提高JS代码的性能

    摘要: 高效使用Set! 作者:前端小智 原文:如何使用 Set 来提高代码的性能 Fundebug经授权转载,版权归原作者所有. 为了保证的可读性,本文采用意译而非直译. 我确信有很多开发人员坚持使 ...

  2. php 高级 提高PHP代码的性能10条建议

    1.echo比print要快很多.两个方法都会在页面上打印东西,不过echo不返回任何值,print会在成功或失败的时候返回0或1. 2.include_once比include更加耗时.因为它需要去 ...

  3. ☕【Java技术指南】「并发编程专题」Fork/Join框架基本使用和原理探究(基础篇)

    前提概述 Java 7开始引入了一种新的Fork/Join线程池,它可以执行一种特殊的任务:把一个大任务拆成多个小任务并行执行. 我们举个例子:如果要计算一个超大数组的和,最简单的做法是用一个循环在一 ...

  4. java Fork/Join框架

    应用程序并行计算遇到的问题 当硬件处理能力不能按摩尔定律垂直发展的时候,选择了水平发展.多核处理器已广泛应用,未来处理器的核心数将进一步发布,甚至达到上百上千的数量.而现在很多的应用程序在运行在多核心 ...

  5. JDK1.7之Fork/join

    Fork/Join框架是Java 7提供的一个用于并行执行任务的框架,是一个把大任务分割成若干个小任务,最终汇总每个小任务结果后得到大任务结果的框架.Fork/Join框架要完成两件事情: 1.任务分 ...

  6. Java并发——Fork/Join框架

    为了防止无良网站的爬虫抓取文章,特此标识,转载请注明文章出处.LaplaceDemon/ShiJiaqi. http://www.cnblogs.com/shijiaqi1066/p/4631466. ...

  7. Java Fork/Join 框架

    简介 从JDK1.7开始,Java提供Fork/Join框架用于并行执行任务,它的思想就是讲一个大任务分割成若干小任务,最终汇总每个小任务的结果得到这个大任务的结果. 这种思想和MapReduce很像 ...

  8. jdk7 并行计算框架Fork/Join

    故名思义,拆分fork+合并join.jdk1.7整合Fork/Join,性能上有大大提升. 思想:充分利用多核CPU把计算拆分成多个子任务,并行计算,提高CPU利用率大大减少运算时间.有点像,Map ...

  9. Java并发——Fork/Join框架与ForkJoinPool

    为了防止无良网站的爬虫抓取文章,特此标识,转载请注明文章出处.LaplaceDemon/ShiJiaqi. http://www.cnblogs.com/shijiaqi1066/p/4631466. ...

随机推荐

  1. Java面向对象编程 -3

    this关键字 this可以算是Java里面比较复杂的关键字,因为this的使用形式约定了它的灵活性,在程序里面,使用this可以实现以下三类结构的描述: 当前类中的属性:this属性; 当前类中的方 ...

  2. Oracle的TO_CHAR()格式化数字为百分数的字符串

    TO_CHAR(-0.926903249,'FM999999990.00') || '%' 说明: 1,(点号) :'99.99' : 点号,不要念为"句号",句号是个圆圈,点号只 ...

  3. Mac 如何导出ipa文件中Assets.car包中的切图

    在之前 获取 AppStore 中 应用 的 IPA 包文件(Mac OS 13+)中获取到应用的 IPA 包,可以取出应用的部分图片(如 Logo),如果项目工程中把图片添加到 Assets.xca ...

  4. iOS 开发之 RunLoop 详解

    1)什么是 Runloop ? 1.字面上是运行循环,内部就是 do-while 循环,在这个循环内不断地处理各种任务. 2.一个线程对应一个 Runloop ,主线程的 RunLoop 默认是开启的 ...

  5. iview table表格内容为数组或者对象的子元素时问题讨论

    正常情况下,iview框架table表格内容只需配置好 key 就OK, 稍微复杂点就是用一个reder函数进行操作(params.row 为本行数据) . 以上问题都很好解决,无需太动脑筋. 开发中 ...

  6. 一份非常值得一看的Java面试题

    包含的模块 本文分为十九个模块,分别是: Java 基础.容器.多线程.反射.对象拷贝.Java Web .异常.网络.设计模式.Spring/Spring MVC.Spring Boot/Sprin ...

  7. Nginx 七层反向代理

    目录 1.代理 2.正向代理 3.反向代理 4.Nginx 反向代理 5.Nginx 反向代理相关指令介绍 ①.listen ②.server_name ③.location ④.proxy_pass ...

  8. python3 getopt用法

    python channel_builder.py -s /Users/graypn/ -d /Users/graypn/Documents -m 7 --out=report/xx.html 参数也 ...

  9. 关于Java大整数是否是素数

    题目描述 请编写程序,从键盘输入两个整数m,n,找出等于或大于m的前n个素数. 输入格式: 第一个整数为m,第二个整数为n:中间使用空格隔开.例如: 103 3 输出格式: 从小到大输出找到的等于或大 ...

  10. cubic-bezier 个人学习理解

    cubic-bezier 三次贝塞尔曲线函数,是一种动画的速度曲线.根据动画设置的时间,速度相应变化 四个点 P0,P1,P2,P3,其中P0是起点,坐标(0,0),P3是终点,坐标(1,1) PI和 ...