测试并发应用(四)监控Fork/Join池

简介:

声明:本文是《 Java 7 Concurrency Cookbook 》的第八章, 作者: Javier Fernández González 译者:郑玉婷

监控Fork/Join池

Executor 框架提供了线程的创建和管理,来实现任务的执行机制。Java 7 包括了一个 Executor 框架的延伸为了一种特别的问题而使用的,将比其他解决方案的表现有所提升(可以直接使用 Thread 对象或者 Executor 框架)。它是 Fork/Join 框架。

此框架是为了解决可以使用 divide 和 conquer 技术,使用 fork() 和 join() 操作把任务分成小块的问题而设计的。主要的类实现这个行为的是 ForkJoinPool 类。

在这个指南,你将学习从ForkJoinPool类可以获取的信息和如何获取这些信息。

准备

指南中的例子是使用Eclipse IDE 来实现的。如果你使用Eclipse 或者其他的IDE,例如NetBeans, 打开并创建一个新的java任务。

怎么做呢…

按照这些步骤来实现下面的例子:

//1.   创建一个类,名为 Task, 扩展 RecursiveAction 类。

public class Task extends RecursiveAction {

	// 2. 声明一个私有 int array 属性,名为 array,用来储存你要增加的 array 的元素。
	private int[] array;

	// 3. 声明2个私有 int 属性,名为 start 和 end,用来储存 此任务已经处理的元素块的头和尾的位置。
	private int start;
	private int end;

	// 4. 实现类的构造函数,初始化属性值。
	public Task(int array[], int start, int end) {
		this.array = array;
		this.start = start;
		this.end = end;
	}

	// 5. 用任务的中心逻辑来实现 compute()
	// 方法。如果此任务已经处理了超过100任务,那么把元素集分成2部分,再创建2个任务分别来执行这些部分,使用 fork() 方法开始执行,并使用
	// join() 方法等待它的终结。
	protected void compute() {
		if (end - start > 100) {
			int mid = (start + end) / 2;
			Task task1 = new Task(array, start, mid);
			Task task2 = new Task(array, mid, end);

			task1.fork();
			task2.fork();

			task1.join();
			task2.join();

			// 6. 如果任务已经处理了100个元素或者更少,那么在每次操作之后让线程进入休眠5毫秒来增加元素。
		} else {
			for (int i = start; i < end; i++) {
				array[i]++;

				try {
					Thread.sleep(5);
				} catch (InterruptedException e) {
					e.printStackTrace();
				}
			}
		}
	}
}

//7. 创建例子的主类通过创建一个类,名为 Main 并添加 main()方法。

import java.util.concurrent.ForkJoinPool;
import java.util.concurrent.TimeUnit;

public class Main {

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

		// 8. 创建 ForkJoinPool 对象,名为 pool。
		ForkJoinPool pool = new ForkJoinPool();

		// 9. 创建 10,000个元素的整数 array ,名为 array。
		int array[] = new int[10000];

		// 10. 创建新的 Task 对象来处理整个array。
		Task task1 = new Task(array, 0, array.length);

		// 11. 使用 execute() 方法 把任务发送到pool里执行。
		pool.execute(task1);

		// 12. 当任务还未结束执行,调用 showLog() 方法来把 ForkJoinPool 类的状态信息写入,然后让线程休眠一秒。
		while (!task1.isDone()) {
			showLog(pool);
			TimeUnit.SECONDS.sleep(1);
		}

		// 13. 使用 shutdown() 方法关闭pool。
		pool.shutdown();

		// 14. 使用 awaitTermination() 方法 等待pool的终结。
		pool.awaitTermination(1, TimeUnit.DAYS);

		// 15. 调用 showLog() 方法写关于 ForkJoinPool 类状态的信息并写一条信息到操控台表明结束程序。
		showLog(pool);
		System.out.printf("Main: End of the program.\n");
	}

	// 16. 实现 showLog() 方法。它接收 ForkJoinPool 对象作为参数和写关于线程和任务的执行的状态的信息。
	private static void showLog(ForkJoinPool pool) {
		System.out.printf("**********************\n");
		System.out.printf("Main: Fork/Join Pool log\n");
		System.out.printf("Main: Fork/Join Pool: Parallelism:%d\n",
				pool.getParallelism());
		System.out.printf("Main: Fork/Join Pool: Pool Size:%d\n",
				pool.getPoolSize());
		System.out.printf("Main: Fork/Join Pool: Active Thread Count:%d\n",
				pool.getActiveThreadCount());
		System.out.printf("Main: Fork/Join Pool: Running Thread Count:%d\n",
				pool.getRunningThreadCount());
		System.out.printf("Main: Fork/Join Pool: Queued Submission:%d\n",
				pool.getQueuedSubmissionCount());
		System.out.printf("Main: Fork/Join Pool: Queued Tasks:%d\n",
				pool.getQueuedTaskCount());
		System.out.printf("Main: Fork/Join Pool: Queued Submissions:%s\n",
				pool.hasQueuedSubmissions());
		System.out.printf("Main: Fork/Join Pool: Steal Count:%d\n",
				pool.getStealCount());
		System.out.printf("Main: Fork/Join Pool: Terminated :%s\n",
				pool.isTerminated());
		System.out.printf("**********************\n");
	}
}

它是如何工作的…

在这个指南,你实现了任务 使用 ForkJoinPool 类和一个扩展 RecursiveAction 类的 Task 类来增加array的元素;它是 ForkJoinPool 类执行的任务种类之一。当任务还在处理array时,你就把关于 ForkJoinPool 类的状态信息打印到操控台。
你使用了ForkJoinPool类中的以下方法:

  • getPoolSize(): 此方法返回 int 值,它是ForkJoinPool内部线程池的worker线程们的数量。
  • getParallelism(): 此方法返回池的并行的级别。
  • getActiveThreadCount(): 此方法返回当前执行任务的线程的数量。
  • getRunningThreadCount():此方法返回没有被任何同步机制阻塞的正在工作的线程。
  • getQueuedSubmissionCount(): 此方法返回已经提交给池还没有开始他们的执行的任务数。
  • getQueuedTaskCount(): 此方法返回已经提交给池已经开始他们的执行的任务数。
  • hasQueuedSubmissions(): 此方法返回 Boolean 值,表明这个池是否有queued任务还没有开始他们的执行。
  • getStealCount(): 此方法返回 long 值,worker 线程已经从另一个线程偷取到的时间数。
  • isTerminated(): 此方法返回 Boolean 值,表明 fork/join 池是否已经完成执行。

参见

第五章,Fork/Join 框架: Fork/Join 池的创建
第七章,自定义并发类:实现 ThreadFactory 接口生成Fork/Join 框架的自定义线程
第七章,自定义并发类:自定义在the Fork/Join框架的运行任务

文章转自 并发编程网-ifeve.com

目录
相关文章
|
3天前
|
敏捷开发 JavaScript 测试技术
深入理解与应用软件测试中的Mock技术
【5月更文挑战第5天】 在现代软件开发过程中,单元测试作为保障代码质量的重要环节,其独立性和可靠性至关重要。Mock技术应运而生,为开发者提供了一种在隔离环境下模拟外部依赖的方法。本文将深入探讨Mock技术的概念、实现方式及其在软件测试中的应用,旨在帮助读者更好地理解和运用这一强大的测试工具,以提升测试效率和软件质量。
|
6天前
|
安全 IDE Java
Java串口通信技术探究2:RXTX库单例测试及应用
Java串口通信技术探究2:RXTX库单例测试及应用
24 4
|
6天前
|
数据采集 机器学习/深度学习 人工智能
探索AI在软件测试中的应用与挑战
【5月更文挑战第2天】本文将探讨人工智能(AI)在软件测试领域的应用及其带来的挑战。我们将详细讨论AI如何改变软件测试的方式,包括自动化测试、预测性测试、智能化缺陷检测等。同时,我们也将探讨AI在软件测试中面临的挑战,如数据质量问题、模型的可解释性、以及对现有测试流程的影响等。
|
7天前
|
Java 测试技术 开发者
深入理解与应用单元测试:软件质量的守护者
【4月更文挑战第30天】 在现代软件开发过程中,单元测试作为保障代码健康的重要环节,其地位日益凸显。本文将探讨单元测试的核心概念、实施单元测试的重要性以及如何高效地设计并执行单元测试。通过实例分析,我们将揭示单元测试在确保软件产品质量和加速开发周期中的关键作用。
|
7天前
|
敏捷开发 测试技术 持续交付
探索自动化测试在敏捷开发中的应用移动应用的未来:跨平台开发与操作系统的融合
【4月更文挑战第30天】随着软件开发周期的不断缩短,传统的软件测试方法逐渐显得力不从心。本文将深入探讨自动化测试在敏捷开发环境中的关键作用,分析其如何提高测试效率、减少人力资源成本,并确保软件产品的质量与稳定性。通过案例分析,我们还将讨论实施自动化测试的最佳实践和面临的挑战,为追求高效敏捷开发的组织提供参考。
|
7天前
|
数据采集 机器学习/深度学习 人工智能
自动化测试中AI辅助技术的应用与挑战
【4月更文挑战第30天】随着人工智能(AI)技术的飞速发展,其在软件自动化测试领域的应用日益增多。本文探讨了AI辅助技术在自动化测试中的应用情况,包括智能化测试用例生成、测试执行监控、缺陷预测及测试结果分析等方面。同时,文章还分析了在融合AI技术时所面临的挑战,如数据质量要求、模型的透明度与解释性问题以及技术整合成本等,并提出了相应的解决策略。
|
7天前
|
机器学习/深度学习 算法 UED
【Python 机器学习专栏】A/B 测试在机器学习项目中的应用
【4月更文挑战第30天】A/B测试在数据驱动的机器学习项目中扮演关键角色,用于评估模型性能、算法改进和特征选择。通过定义目标、划分群组、实施处理、收集数据和分析结果,A/B测试能帮助优化模型和用户体验。Python提供工具如pandas和scipy.stats支持实验实施与分析。注意样本量、随机性、时间因素和多变量分析,确保测试有效性。A/B测试助力于持续改进机器学习项目,实现更好的成果。
|
8天前
|
前端开发 IDE 数据可视化
深入理解与应用自动化测试框架Selenium的最佳实践
【4月更文挑战第30天】 本文将深入剖析自动化测试框架Selenium的核心原理,并结合最佳实践案例,探讨如何有效提升测试覆盖率和效率。文中不仅涉及Selenium的架构解析,还将提供针对性的策略来优化测试脚本,确保测试流程的稳定性与可靠性。通过实例演示,读者可以掌握如何在不同测试场景中灵活运用Selenium,以及如何处理常见的技术挑战。
|
8天前
|
缓存 监控 前端开发
【Flutter前端技术开发专栏】Flutter应用的性能调优与测试
【4月更文挑战第30天】本文探讨了Flutter应用的性能调优策略和测试方法。性能调优对提升用户体验、降低能耗和增强稳定性至关重要。优化布局(避免复杂嵌套,使用`const`构造函数)、管理内存、优化动画、实现懒加载和按需加载,以及利用Flutter的性能工具(如DevTools)都是有效的调优手段。性能测试包括基准测试、性能分析、压力测试和电池效率测试。文中还以ListView为例,展示了如何实践这些优化技巧。持续的性能调优是提升Flutter应用质量的关键。
【Flutter前端技术开发专栏】Flutter应用的性能调优与测试
|
8天前
|
敏捷开发 监控 前端开发
深入理解与应用自动化测试框架:以Selenium为例
【4月更文挑战第30天】 在软件开发的快速迭代周期中,质量保证(QA)团队面临持续的压力,需确保产品在每次发布时都达到预期的质量标准。为了应对这一挑战,自动化测试成为了关键工具,它不仅提高了测试效率,还确保了测试的一致性和可重复性。本文将探讨自动化测试框架Selenium的核心组件、工作原理及其在实际测试中的应用。通过分析Selenium的优势和面临的常见问题,我们将讨论如何有效地集成Selenium到现有的测试流程中,以及如何克服常见的技术障碍。我们的目标是为读者提供一个清晰的指南,帮助他们理解和利用自动化测试框架来优化他们的软件测试实践。

热门文章

最新文章