线程的创建与使用(二)

上篇介绍了创建线程的前两种方式,继承Thread和实现Runnable接口,但它两都有个天生的缺陷,就是没有返回值。执行了半天没有返回值,这可咋整?


其实,如果要有返回值也是有方法的。java提供了Callable接口、Future接口、FutureTask接口,通过使用它们就能在线程执行完得到返回结果,话不多说,试一把就知道了

public class Test3 {
    public static void main(String[] args) {
        //创建一个有5个固定线程的线程池
        ExecutorService executorService = Executors.newFixedThreadPool(5);
        try {
            Future run = executorService.submit(new Runnable() {
                @Override
                public void run() {
                    System.out.println("哈哈哈哈哈哈");
                }
            });
            System.out.println("run的返回值:"+run.get());
        }catch (Exception e) {
            e.printStackTrace();
        }
        executorService.shutdown();
    }
}

执行结果:Runnable无返回值

Runnable
public class Test3 {
    public static void main(String[] args) {
        //创建一个有5个固定线程的线程池
        ExecutorService executorService = Executors.newFixedThreadPool(5);
        try {
            Future<String> future = executorService.submit(new Callable<String>() {
                @Override
                public String call() throws Exception {
                    System.out.println("呵呵呵呵呵");
                    return "呵呵呵呵呵";
                }
            });
            System.out.println("future的执行结果:"+future.get());

        }catch (Exception e) {
            e.printStackTrace();
        }
        executorService.shutdown();
    }
}

执行结果:Callable是有返回值的


Callable

上面的两个小例子可以就看出Runnable和Callable的本质区别就是有无返回值,其中使用了Future接口,ExecutorService接口,下面介绍下这些接口:

Callable

public interface Callable<V> {
    V call() throws Exception;
}

接口Callable中只声明了一个方法call(),返回类型就是V,当然光使用一个Callable还不能获取返回值,还需要ExecutorService、Future一起使用(Callable、ExecutorService、Future均属于JUC包下,而Runnable属于java.lang包下)

public interface ExecutorService extends Executor {
    //方法省略
}

Future

public interface Future<V> {
   
    boolean cancel(boolean mayInterruptIfRunning);

    boolean isCancelled();

    boolean isDone();
    
    V get() throws InterruptedException, ExecutionException;
    
    V get(long timeout, TimeUnit unit)
        throws InterruptedException, ExecutionException, TimeoutException;
}

Future 表示异步计算的结果。它提供了检查计算是否完成的方法,以等待计算的完成,并获取计算的结果。计算完成后只能使用 get 方法来获取结果,如有必要,计算完成前可以阻塞此方法。

方法介绍:

cancel用来取消任务,如果取消成功返回true,取消失败返回false,mayInterruptIfRunning的意思是是否取消执行一半没有执行完毕的任务。若mayInterruptIfRunning设为true,则表示能够强制取消执行一半的任务,如果被取消任务还没完成,返回true,如果被取消任务已完成,返回false;若mayInterruptIfRunning设为false,若果任务没完成,返回false,如果任务已完成,返回false(任务完成,不管mayInterruptIfRunning设成啥,都返回false;任务没开始,不管mayInterruptIfRunning设成啥,都返回true)。
isCancelled用来表示任务是否被成功取消,如果成功,返回true。
isDone用来表示任务是否成功完成,如果成功,返回true。
get()用来获取返回值,这是个阻塞方法,会等任务执行完毕再返回。
**get(long timeout, TimeUnit unit)也是用来获取返回值,比上一个聪明点,如果在设定时间内任务还没完成,就返回null。

public class Test3 {
    public static void main(String[] args) {
        //创建一个有5个固定线程的线程池
        ExecutorService executorService = Executors.newFixedThreadPool(5);
        try {
            Future<String> future2 = executorService.submit(new Callable<String>() {
                @Override
                public String call() throws Exception {
                    try {
                        while(true){
                            System.out.println("task2 running");
                            Thread.sleep(500);
                        }
                    }catch (Exception e){
                        System.out.println("Interrupted task2.");
                    }
                    return "嘻嘻嘻嘻嘻";
                }
            });
            Thread.sleep(3000);
            System.out.println("task2 cancel:" + future2.cancel(true));
        }catch (Exception e){
            e.printStackTrace();
        }
        executorService.shutdown();
    }
}

可以看到任务future2于主线程异步执行,主线程sleep3秒钟,停止任务,停止任务成功。

FutureTask

FutureTask<V>实现了RunnableFuture接口,而RunnableFuture又继承自
Runnable,Future<V>,也就是时候FutureTask既实现了Runnable又实现了Future<V>,不仅能通过Thread包装执行,还能提交给ExecutorService执行,并通过get()来获取返回结果。

有两个构造方法:

public class Test4 {
    public static void main(String[] args) {
        Random random = new Random();
        Callable<Integer> task = new Callable<Integer>() {
            @Override
            public Integer call() throws Exception {
                return random.nextInt(100);
            }
        };
        FutureTask<Integer> futureTask1 = new FutureTask<Integer>(task);
        Thread thread = new Thread(futureTask1);
        thread.start();
        try {
            System.out.println("用Thread执行的结果为:"+futureTask1.get());
            System.out.println("下面准备使用Executors执行");
            Thread.sleep(1000);
        } catch (Exception e) {
            e.printStackTrace();
        }
        FutureTask<Integer> futureTask2 = new FutureTask<Integer>(task);
        ExecutorService executorService = Executors.newFixedThreadPool(2);
        executorService.submit(futureTask2);
        try {
            System.out.println("用Executors执行的结果为:"+futureTask2.get());
        } catch (Exception e) {
            e.printStackTrace();
        }
        executorService.shutdown();
    }
}

小结

此篇主要介绍了使用Callable接口创建有返回结果的线程,其中涉及到了Future、FutureTask、ExecutorService等接口的使用。Future单词本身含义为将来,在程序中的意思应该就是提交一个请求后,不用傻等着结果返回,可以先做别的操作,等返回结果真正处理完成并返回给请求方后,请求方再做相应处理。

略陈固陋,如有不当之处,欢迎各位看官批评指正!

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 194,911评论 5 460
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 82,014评论 2 371
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 142,129评论 0 320
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 52,283评论 1 264
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 61,159评论 4 357
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 46,161评论 1 272
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 36,565评论 3 382
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 35,251评论 0 253
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 39,531评论 1 292
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 34,619评论 2 310
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 36,383评论 1 326
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 32,255评论 3 313
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 37,624评论 3 299
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 28,916评论 0 17
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,199评论 1 250
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 41,553评论 2 342
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 40,756评论 2 335

推荐阅读更多精彩内容