美文网首页
线程的创建与使用(二)

线程的创建与使用(二)

作者: 少博先生 | 来源:发表于2017-09-13 23:26 被阅读0次

上篇介绍了创建线程的前两种方式,继承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单词本身含义为将来,在程序中的意思应该就是提交一个请求后,不用傻等着结果返回,可以先做别的操作,等返回结果真正处理完成并返回给请求方后,请求方再做相应处理。

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

相关文章

  • Linux下多线程的使用

    一、线程的创建 1.创建分离线程方法一:(常用) 方法二: 2.创建非分离线程 二、线程的使用 三、分离线程与非分...

  • 线程的创建与使用(二)

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

  • 线程的创建销毁与退出,线程属性,互斥锁,读写锁

    一、线程的创建与退出 1.1 线程的创建 主线程 子线程 1.2 线程的退出 线程的退出可以使用return的方式...

  • iOS学习笔记_NSThread基本使用

    一、NSThread基本使用 二、 NSThread的创建 1.创建线程(需要手动启动线程) 2.分离子线程,自动...

  • thread

    使用 spawn 创建新线程 使用 join 等待所有线程结束 将handle.join()提前 线程与 move...

  • POSIX多线程—线程基本概念

    1.线程建立与使用 创建线程 初始线程 线程分离 2.线程生命周期 就绪态...

  • Python 多线程笔记

    Python 多线程笔记 创建线程 1. 使用函数创建多线程 2. 使用类创建多线程 继承 Thread 类 重写...

  • JAVA进程和线程-day01

    A 进程 线程 线程和进程的区别 线程使用的场合 创建线程的方法 并发原理 使用Runnable创建并启动线程 线...

  • NSThread

    大神连接 创建使用线程 第一种方式 创建手动启动线程 第二种 创建完成自动执行线程 第三种 相关方法

  • Python 线程池原理及实现

    概述 传统多线程方案会使用“即时创建, 即时销毁”的策略。尽管与创建进程相比,创建线程的时间已经大大的缩短,但是如...

网友评论

      本文标题:线程的创建与使用(二)

      本文链接:https://www.haomeiwen.com/subject/qwdhsxtx.html