首页 > 解决方案 > 提交带有回调的可调用任务时 ExecutorService 的工作原理

问题描述

如何在 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() 为空呢?

标签: javamultithreadingcallbackexecutorservice

解决方案


return e1.submit(() -> {
    try {
        if (flag) {
            return;
        }
        taskToRun.call();
    } catch (Exception e) {
        System.out.println("garbage ---");
    }
});

在这个片段中,传递给submit方法 on的 lambdaExecutorService e1被解释为 a Runnable,这就是为什么返回值为null. 查看此文档以获取更多详细信息。您需要将可调用对象taskToRun本身传递e1.submit()给能够获取可调用对象的返回值。此外,将逻辑放在Callable自身内部会更干净。

我已经重写了示例代码来满足您的要求。

package stackoverflow.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;
  }
}

推荐阅读