提交带有回调的可调用任务时 ExecutorService 的工作原理
How ExecutorService works when submit a callable task with callback
如何在 executorService.submit() 回调中添加自定义检查。
只要标志为真,我希望我的线程不做任何事情。如果标志为假,那么它应该照常工作。
import org.apache.commons.lang3.concurrent.BasicThreadFactory;
import java.util.concurrent.Callable;
import java.util.concurrent.Future;
import java.util.concurrent.ExecutionException;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
public class test{
private static volatile boolean flag= false;
public static void main(String[] args) throws ExecutionException, InterruptedException {
Callable<Boolean> callableTask = () -> {
return isFlag();
};
Future a = submitAsync(callableTask);
System.out.println("tartest"+a.get());
}
public static Future submitAsync(Callable taskToRun) {
ExecutorService e1 = Executors.newSingleThreadExecutor(new BasicThreadFactory.Builder().namingPattern("").build());
return e1.submit(() -> {
try {
if (flag) {
return;
}
taskToRun.call();
} catch (Exception e) {
System.out.println("garbage ---");
}
});
// return e1.submit(taskToRun);
}
public static boolean isFlag() {
return true;
}
}
此处,a.get() 返回 null。如果我用 e1.submit(taskToRun) 替换 e1.submit(....),它在给定的代码片段中被注释,那么它工作正常。那么为什么 a.get() 为空?
您的 submitAsync
方法 return 是调用 ExecutorService.submit(Runnable) 的结果的 Future。来自 the documentation of that method:
Submits a Runnable task for execution and returns a Future representing that task. The Future's get method will return null upon successful completion.
如果您想 return 一个产生值的 Future,请将 return 语句添加到您的 lambda 中,这样它将被解释为 Callable 而不是 Runnable:
return taskToRun.call();
} catch (Exception e) {
e.printStackTrace();
return null;
}
return e1.submit(() -> {
try {
if (flag) {
return;
}
taskToRun.call();
} catch (Exception e) {
System.out.println("garbage ---");
}
});
在此代码段中,传递给 ExecutorService e1
上的 submit
方法的 lambda 被解释为 Runnable
,这就是为什么 return 值为 null
.查看此 documentation 了解更多详情。您需要将可调用对象 taskToRun
本身传递给 e1.submit()
才能获得可调用对象的 return 值。此外,将逻辑放在 Callable
本身内会更清晰。
我已经重写了示例代码以满足您的要求。
package Whosebug.test;
import org.apache.commons.lang3.concurrent.BasicThreadFactory;
import java.util.concurrent.Callable;
import java.util.concurrent.Future;
import java.util.concurrent.ExecutionException;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
public class test{
private static volatile boolean flag = true;
public static void main(String[] args) throws ExecutionException, InterruptedException {
Callable<Boolean> callableTask = () -> {
if (flag) {
System.out.println("Flag is true, returning without executing callable logic !");
return false;
}
System.out.println("Flag is false, executing callable logic !");
return isFlag();
};
Future a = submitAsync(callableTask);
System.out.println("test " + a.get());
}
private static Future<Boolean> submitAsync(Callable<Boolean> taskToRun) {
ExecutorService e1 = Executors.newSingleThreadExecutor(new BasicThreadFactory.Builder().namingPattern("").build());
return e1.submit(taskToRun);
}
private static boolean isFlag() {
return true;
}
}
如何在 executorService.submit() 回调中添加自定义检查。 只要标志为真,我希望我的线程不做任何事情。如果标志为假,那么它应该照常工作。
import org.apache.commons.lang3.concurrent.BasicThreadFactory;
import java.util.concurrent.Callable;
import java.util.concurrent.Future;
import java.util.concurrent.ExecutionException;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
public class test{
private static volatile boolean flag= false;
public static void main(String[] args) throws ExecutionException, InterruptedException {
Callable<Boolean> callableTask = () -> {
return isFlag();
};
Future a = submitAsync(callableTask);
System.out.println("tartest"+a.get());
}
public static Future submitAsync(Callable taskToRun) {
ExecutorService e1 = Executors.newSingleThreadExecutor(new BasicThreadFactory.Builder().namingPattern("").build());
return e1.submit(() -> {
try {
if (flag) {
return;
}
taskToRun.call();
} catch (Exception e) {
System.out.println("garbage ---");
}
});
// return e1.submit(taskToRun);
}
public static boolean isFlag() {
return true;
}
}
此处,a.get() 返回 null。如果我用 e1.submit(taskToRun) 替换 e1.submit(....),它在给定的代码片段中被注释,那么它工作正常。那么为什么 a.get() 为空?
您的 submitAsync
方法 return 是调用 ExecutorService.submit(Runnable) 的结果的 Future。来自 the documentation of that method:
Submits a Runnable task for execution and returns a Future representing that task. The Future's get method will return null upon successful completion.
如果您想 return 一个产生值的 Future,请将 return 语句添加到您的 lambda 中,这样它将被解释为 Callable 而不是 Runnable:
return taskToRun.call();
} catch (Exception e) {
e.printStackTrace();
return null;
}
return e1.submit(() -> {
try {
if (flag) {
return;
}
taskToRun.call();
} catch (Exception e) {
System.out.println("garbage ---");
}
});
在此代码段中,传递给 ExecutorService e1
上的 submit
方法的 lambda 被解释为 Runnable
,这就是为什么 return 值为 null
.查看此 documentation 了解更多详情。您需要将可调用对象 taskToRun
本身传递给 e1.submit()
才能获得可调用对象的 return 值。此外,将逻辑放在 Callable
本身内会更清晰。
我已经重写了示例代码以满足您的要求。
package Whosebug.test;
import org.apache.commons.lang3.concurrent.BasicThreadFactory;
import java.util.concurrent.Callable;
import java.util.concurrent.Future;
import java.util.concurrent.ExecutionException;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
public class test{
private static volatile boolean flag = true;
public static void main(String[] args) throws ExecutionException, InterruptedException {
Callable<Boolean> callableTask = () -> {
if (flag) {
System.out.println("Flag is true, returning without executing callable logic !");
return false;
}
System.out.println("Flag is false, executing callable logic !");
return isFlag();
};
Future a = submitAsync(callableTask);
System.out.println("test " + a.get());
}
private static Future<Boolean> submitAsync(Callable<Boolean> taskToRun) {
ExecutorService e1 = Executors.newSingleThreadExecutor(new BasicThreadFactory.Builder().namingPattern("").build());
return e1.submit(taskToRun);
}
private static boolean isFlag() {
return true;
}
}