- Java 双重比较
- java - 比较器与 Apache BeanComparator
- Objective-C 完成 block 导致额外的方法调用?
- database - RESTful URI 是否应该公开数据库主键?
我正在阅读 Java ForkJoin 框架。不直接在 ForkJoinTask
的实现上调用 invoke()
(例如 RecursiveTask
),而是实例化 ForkJoinPool 有什么额外的好处
并调用 pool.invoke(task)
?当我们将这 2 个方法都称为 invoke
时究竟会发生什么?
从源代码来看,似乎如果 recursiveTask.invoke
被调用,它将调用它的 exec
并最终 compute
,在一个托管线程池方式。因此,更令人困惑的是,为什么我们使用成语 pool.invoke(task)
。
我写了一些简单的代码来测试性能差异,但我没有看到任何差异。也许测试代码是错误的?见下文:
public class MyForkJoinTask extends RecursiveAction {
private static int totalWorkInMillis = 20000;
protected static int sThreshold = 1000;
private int workInMillis;
public MyForkJoinTask(int work) {
this.workInMillis = work;
}
// Average pixels from source, write results into destination.
protected void computeDirectly() {
try {
ForkJoinTask<Object> objectForkJoinTask = new ForkJoinTask<>();
Thread.sleep(workInMillis);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
@Override
protected void compute() {
if (workInMillis < sThreshold) {
computeDirectly();
return;
}
int discountedWork = (int) (workInMillis * 0.9);
int split = discountedWork / 2;
invokeAll(new MyForkJoinTask(split),
new MyForkJoinTask(split));
}
public static void main(String[] args) throws Exception {
System.out.printf("Total work is %d in millis.%n", totalWorkInMillis);
System.out.printf("Threshold is %d in millis.%n", sThreshold);
int processors = Runtime.getRuntime().availableProcessors();
System.out.println(Integer.toString(processors) + " processor"
+ (processors != 1 ? "s are " : " is ")
+ "available");
MyForkJoinTask fb = new MyForkJoinTask(totalWorkInMillis);
ForkJoinPool pool = new ForkJoinPool();
long startTime = System.currentTimeMillis();
// These 2 seems no difference!
pool.invoke(fb);
// fb.compute();
long endTime = System.currentTimeMillis();
System.out.println("Took " + (endTime - startTime) +
" milliseconds.");
}
}
最佳答案
RecursiveTask
类的compute()
方法只是一个包含任务代码的抽象方法。它不使用池中的新线程,如果您正常调用它,它也不会在池管理的线程中运行。
fork join pool 上的 invoke
方法向池提交任务,然后池开始在单独的线程上运行,调用该线程上的 compute
方法,然后然后等待结果。
您可以在 RecursiveTask 和 ForkJoinPool 的 java 文档中的措辞中看到这一点。 invoke()
方法实际执行任务,而 compute()
方法只是封装计算。
protected abstract V compute()
The main computation performed by this task.
和ForkJoinPool
public <T> T invoke(ForkJoinTask<T> task)
Performs the given task, returning its result upon completion. ...
因此,对于计算方法,您正在做的是在 fork 连接池之外运行对 compute
的第一次调用。您可以通过在计算方法中添加日志行来对此进行测试。
System.out.println(this.inForkJoinPool());
您还可以通过记录线程 ID 来检查它是否在同一个线程中运行
System.out.println(Thread.currentThread().getId());
一旦您调用了 invokeAll
,该调用中包含的子任务就会在池中运行。但请注意,它不一定在调用 compute()
之前创建的池中运行。您可以注释掉您的 new ForkJoinPool()
代码,它仍然会运行。有趣的是,java 7 文档说如果在池管理线程之外调用 invokeAll()
方法将抛出异常,但 java 8 文档不会。请注意,我没有在 Java 7 中测试过它(只有 8 个)。但很有可能,当在 Java 7 中直接调用 compute()
时,您的代码会抛出异常。
两个结果同时返回的原因是毫秒不够准确,无法记录在池管理线程中启动第一个线程或仅运行第一个 compute
调用的区别在现有线程中。
Sierra 和 Bates 的 OCA/OCP 学习指南建议您使用 fork join 框架的方式是从池中调用 invoke()
。它明确了你使用的是哪个池,也意味着你可以将多个任务提交到同一个池中,这样就节省了每次重新创建新池的开销。从逻辑上讲,将所有任务计算保存在池管理的线程中(或者至少我认为是这样)也更清晰。
pool.invoke()
调用在特定池上调用;当第一次调用 task.invoke
或 task.invokeAll
时,而不是让框架自行创建。这意味着您可以将池重新用于新任务,并在创建池时指定 Activity 线程数等内容。这就是区别。将这些日志行添加到您的代码中,尝试一下,您就会看到它在做什么
关于java - ForkJoinPool.invoke() 和 ForkJoinTask.invoke() 或 compute(),我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/34132326/
我有以下代码: Public Delegate Sub SetStatusBarTextDelegate(ByVal StatusText As String) Private Sub SetStat
在调用 Invoke-RestMethod 时使用 Powershell,例如: Invoke-RestMethod -Method Get -Uri "https://google.com/api/
我正在尝试将 Winform 应用程序转换为控制台应用程序。 Winform 应用程序有一个委托(delegate)处理程序。如何在 console 应用程序中编写相同的功能? this.Invoke
在 WPF 中,Dispatcher.Invoke 和直接在控件实例上调用的 Invoke 有什么区别。据我了解,调度程序负责处理线程的消息,Control.Invoke 是否会继续调用 Dispat
我正在研究性能监控系统,它可以将其例程注入(inject)现有程序集。为此,我试图了解 dalvik 代码的工作原理。 下面是我要完成的工作的示例。输入类可能如下所示: class MyClass{
我正在使用 powershell 命令来执行脚本和 cmdlet。因此,在执行 cmdlet 时,我使用了 powershell.invoke,而在执行脚本时,我使用了 pipeline.invoke
有人能解释一下 Invoke-Expression $test 之间的区别吗?和 Invoke-Expression -Command $test ? 变量测试是: $test = "notepad
我有四个类,即 MapperOne、ReducerOne、MapperTwo、ReducerTwo。我想要其中的一个链。 MapperOne-->ReducerOne-->输出文件生成,输入到Mapp
我正在阅读 Java ForkJoin 框架。不直接在 ForkJoinTask 的实现上调用 invoke()(例如 RecursiveTask),而是实例化 ForkJoinPool 有什么额外的
我在调用 Invoke-SqlCmd 时遇到问题,因为它包含第二个 Invoke-SqlCmd 调用: function Get-Foo { $query=` @" WITH data AS (
有人知道如何解决这个问题吗?我创建了一个客户端来使用网络服务。客户端代码为: package cliente; import java.util.List; import handler.Header
我希望使用 P/Invoke 来允许我的 C# 程序集与 native C 库互操作;这需要是跨平台的(即 Mono),因此不能使用混合模式程序集。我想知道使用不安全的 P/invoke 调用并在不安
一般来说,我对使用 Invoke-RestMethod/Invoke-WebRequest 比较陌生 - 我认为这是以下问题的重要背景。 我正在调用如下电话: $Headers = @{ "A
在 Jenkins 的一个自由风格项目(不是说 Maven2/3 项目)中,我有两个可能的构建步骤: 调用 Maven 3 调用顶级 Maven 目标 在不同的安装中,我有不同的组合(有些两者都有,有
这是完整的错误: e: C:\Users\HP\AndroidStudioProjects\MoneyManager\app\src\main\java\com\cruxrepublic\moneym
我正在编写 jQuery 插件并将它们与 AJAX 集成。我正在减少脂肪并专注于基础知识: (function($) { function MyPlugin(el, options) {
有人可以建议我如何处理这条消息吗? CA1060 Move P/Invokes to NativeMethods class Because it is a P/Invoke method, 'UCo
在java中我们可以“用类名调用一个静态方法”也可以“用一个对象调用一个静态方法”java中“用类名调用静态方法”和“用对象调用静态方法”有什么区别? 最佳答案 没有区别,但建议以静态方式调用 sta
尝试从对话框中的 EditText 获取 Edit Text 的值,但一次又一次地出现此错误 Attempt to invoke virtual method 'android.text.Editab
我正在开发一款扑翼应用程序。在出现此错误之前,读取和写入FireStore数据库没有任何问题,但随后突然出现错误(如下所示),并阻止我读取或写入数据库。我一直在寻找答案,但不幸的是,我找不到任何可以解
我是一名优秀的程序员,十分优秀!