转载

Callable - 理解Java的Future模式

简介:

在Java中,创建线程一般有两种方式,一种是继承Thread类,一种是实现Runnable接口。然而,这两种方式的缺点是在线程任务执行结束后,无法获取执行结果。我们一般只能采用共享变量或共享存储以及线程通信的方式实现获得任务结果的目的;

不过,在Java中,也提供了使用Callable和Future来实现获取任务结果的操作。Callable用来执行任务,产生结果,而Future用来获得结果;

@FunctionalInterface
public interface Callable<V> {
    /**
     * Computes a result, or throws an exception if unable to do so.
     *
     * @return computed result
     * @throws Exception if unable to compute a result
     */
    V call() throws Exception;
}

线程实现方式:

1.继承Thread类

2.实现Runnable接口

3.线程池

4.Callable

无论使用继承Thread类还是实现Runnable接口,还是使用线程池都没有办法解决2个问题

1.线程执行没有返回值结果

2.线程执行没有办法抛出异常,只能自己通过try-catch解决

Callable和Runnable类似,在JUC包下,主要区别在于Callable中的call方法可以带返回值并且可以抛出异常

如果需要执行Callable,需要Future实现类的支持,能够接受返回值结果,FutureTask是Future实现类

Future模式

Future莫斯的核心在于:去除了函数的等待时间,并使得原来需要等待的时间段可以用于处理其他业务逻辑;

Future模式:对于多线程,如果线程A要等待线程B的结果,那么线程A没有必要等待线程B,知道线程B有结果,可以先拿到一个未来的Future,等线程B有结果时再取真实的结果;

public class MyCallable implements Callable {
    @Override
    public Object call() throws Exception {
        System.out.println("callable接口中重写的call方法,可以有返回值并且抛出异常!!!");
        return "callable";
    }

    //方案一
    public static void main(String[] args) throws ExecutionException, InterruptedException {
        MyCallable myCallable = new MyCallable();
        //利用FutureTask执行callable并且接受结构
        FutureTask<String> stringFutureTask = new FutureTask<String>(myCallable);
        //利用线程执行task任务
        new Thread(stringFutureTask).start();
        //接受结果,get方法会发生阻塞情况
        System.out.println(stringFutureTask.get());
        System.out.println("mycallable执行完毕!");
    }

}
public class MyCallable implements Callable {
    @Override
    public Object call() throws Exception {
        System.out.println("callable接口中重写的call方法,可以有返回值并且抛出异常!!!");
        return "callable";
    }

    //方案二:submit(Callable task)
    public static void main(String[] args) throws ExecutionException, InterruptedException {
        MyCallable myCallable = new MyCallable();
        //创建一个线程
        ExecutorService executorService = Executors.newFixedThreadPool(3);
        //创建线程执行任务,接受任务结果
        Future submit = executorService.submit(myCallable);
        //接受返回值,get方法会阻塞当前线程
        System.out.println(submit.get());
        System.out.println("利用线程池执行mycallable,完毕!!!");
        //停止
        executorService.shutdown();
    }
}

常用方法:

V get():获取异步执行的结果,如果没有结果可用,此方法会阻塞知道异步计算完成;

V get(Long timeout,TimeUnit unit):获取异步执行结果,如果没哟结果可用,此方法会阻塞,但是会有时间限制,如果阻塞时间超过设定的timeout时间,该方法将抛出异常;

boolean isDone():如果任务执行结束,无论是正常结束或是中途取消还是发生异常,都返回true;

boolean isCanceller():如果任务完成前被取消,则返回true;

boolean cancel(boolean mayInterrupRunning):如果任务还没有开始,执行cancel方法将返回false;如果任务已经启动,执行cancel方法将以中断执行此任务线程的方式来试图停止任务,如果停止成功,返回true;

当任务已经启动,执行cancel(false)方法将不会对正在执行的任务线程产生影响(让线程正常执行到完成),此时返回false;

当任务已经启动,执行cancel方法将返回false,MayInterruptRunning参数表示是否中断执行中的线程;

实际上Future提供了三种功能:

1.能够中断执行中的任务;

2.判断任务是否执行完成;

3.获取任务执行完成后的结果;

get()方法的阻塞性

public class FutureGet {
    public static void main(String[] args) throws ExecutionException, InterruptedException {
        ExecutorService executor=Executors.newFixedThreadPool(2);
        //创建一个Callable,三秒返回String类型
        Callable callable = new Callable() {
            @Override
            public Object call() throws Exception {
                Thread.sleep(3000);
                System.out.println("callable方法执行!");
                return "callable";
            }
        };
        System.out.println("提交任务之前:"+getStringDate());
        Future future = executor.submit(callable);
        System.out.println("提供任务之后,获取结果之前:"+getStringDate());
        System.out.println("获取返回值:"+future.get());
        System.out.println("获取到结果之后:"+getStringDate());
    }

    public static String getStringDate(){
        Date date = new Date();
        SimpleDateFormat format = new SimpleDateFormat("HH:mm:ss");
        String dataString = format.format(date);
        return dataString;
    }
}

通过上面的输出可以看到,在调动submit提交任务之后,主线程本来是继续运行到future.get()的时候就阻塞了,一直等到任务执行完毕,拿到了返回的返回值,主线程才会继续运行;

阻塞性是因为调用了get()方法时,任务没有执行完毕,所以会一直等到任务完成,形成了阻塞;

任务是在调用submit方法时就开始执行了,如果在调用get方法时,任务已经执行完毕,那么就不会造成阻塞;

下面调用方法前先休眠四秒,这时能马上得到返回值:

submit(Runnable task)

因为Runnable是没有返回值,所以如果submit一个Runnable的话,get得到的值肯定为null;

public class SubmitRunnable {
    public static void main(String[] args) throws ExecutionException, InterruptedException {
        ExecutorService executor= Executors.newFixedThreadPool(2);
        Runnable runnable = new Runnable() {
            @Override
            public void run() {
                try {
                    Thread.sleep(2000);
                    System.out.println(Thread.currentThread().getName());
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            }
        };
        Future future = executor.submit(runnable);
        System.out.println("获得返回值:"+future.get());
    }
}

submit(Runnable task,T result)

虽然传入Runnable不能直接返回内容,但是可以通过submit(Runnable task,T result)传入一个载体,通过这个载体获取返回值;

public class Data {
    String name;
    String sex;

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public String getSex() {
        return sex;
    }

    public void setSex(String sex) {
        this.sex = sex;
    }
}

在Runnable的构造方法中传入:

public class MyThreadData implements Runnable {
    Data data;

    public MyThreadData(Data data) {
        this.data = data;
    }

    @Override
    public void run() {
        try {
            Thread.sleep(2000);
            System.out.println("线程执行:");
            data.setName("张三");
            data.setSex("女");
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
    }
    public static void main(String[] arg) throws ExecutionException, InterruptedException {
        ExecutorService executor= Executors.newFixedThreadPool(2);
        Data data = new Data();
        Future<Data> submit = executor.submit(new MyThreadData(data), data);
        System.out.println("返回结果:"+submit.get().getName()+",sex:"+submit.get().getSex());
    }
}

get(long var1,TimeUnit var3)

public class GetTime {
    public static void main(String[] args) throws ExecutionException, InterruptedException, TimeoutException {
        ExecutorService executor= Executors.newFixedThreadPool(2);
        //创建一个Callable,三秒返回String类型
        Callable callable = new Callable() {
            @Override
            public Object call() throws Exception {
                Thread.sleep(3000);
                System.out.println("callable方法执行!");
                return "callable";
            }
        };
        System.out.println("提交任务之前:"+getStringDate());
        Future future = executor.submit(callable);
        System.out.println("提供任务之后,获取结果之前:"+getStringDate());
        System.out.println("获取返回值:"+future.get(2,TimeUnit.SECONDS));
        System.out.println("获取到结果之后:"+getStringDate());
    }

    public static String getStringDate(){
        Date date = new Date();
        SimpleDateFormat format = new SimpleDateFormat("HH:mm:ss");
        String dataString = format.format(date);
        return dataString;
    }
}

更多Java相关信息见 Java 专题页面 https://www.linuxidc.com/topicnews.aspx?tid=19

Linux公社的RSS地址 : https://www.linuxidc.com/rssFeed.aspx

本文永久更新链接地址: https://www.linuxidc.com/Linux/2020-03/162766.htm

原文  https://www.linuxidc.com/Linux/2020-03/162766.htm
正文到此结束
Loading...