如果有人问题你,多线程可以有返回值吗?你怎么回答?

看下面例子,我定义了一个类实现了Callable 接口

  1. public class MyCallable implements Callable<Object> {
  2. @Override
  3. public Object call() throws Exception {
  4. int i=10;
  5. Thread.sleep(10000);
  6. return i;
  7. }
  8. }

Callable如果要想得到返回值需要通过一个叫FutureTask的类帮助。简单的做法是这样的

  1. public void test1() {
  2. try {
  3. FutureTask<Object> task = new FutureTask<Object>(new MyCallable());
  4. new Thread(task).start();
  5. System.out.println("task return : " + task.get().toString());
  6. //get方法会一直阻塞,直到这个线程也就是call方法执行完毕
  7. //可以通过调用isDone()来异步的询问是否已经完成。
  8. } catch (InterruptedException e) {
  9. e.printStackTrace();
  10. } catch (ExecutionException e) {
  11. e.printStackTrace();
  12. }
  13. }

还有另外一种方式可以使用

  1. public void test2(){
  2. ExecutorService newCachedThreadPool = Executors.newCachedThreadPool();
  3. Future<Object> future = newCachedThreadPool.submit(new MyCallable());
  4. try {
  5. //同样可以通过future.isDone()来异步的知道线程是否已经处理完毕
  6. System.out.println(future.get().toString());
  7. }catch (Exception e) {
  8. e.printStackTrace();
  9. } finally {
  10. newCachedThreadPool.shutdown();
  11. }
  12. }

通过上面的例子,我们知道了可以通过callable得到一个线程的返回值了,那么使用callable算多线程吗?

其实第一个例子的一行代码已经暴露出他有多线程的特性了

  1. new Thread(task).start();

看看这个task是什么的东西

  1. public class FutureTask implements RunnableFuture
  2. //是RunnableFuture的一个实现
  3. public interface RunnableFuture extends Runnable, Future
  4. //是Runnable 的子接口,这就是为什么可以放入Thread类中
  5. // 同时他又有future的特性,得到返回值

为了彻底看清Future到底有什么,下面接口就是Future的完整定义了

  1. public interface Future
  2. {
  3. public abstract boolean cancel(boolean flag);
  4. public abstract boolean isCancelled();
  5. public abstract boolean isDone();
  6. public abstract Object get()
  7. throws InterruptedException, ExecutionException;
  8. public abstract Object get(long l, TimeUnit timeunit)
  9. throws InterruptedException, ExecutionException, TimeoutException;
  10. }

现在知道了Callable是借助Runnable来实现多线程的功能,所以说它是多线程也说的过去。那么它是如何获得返回值的呢?

我们都知道线程的启动方法是start方法,然后线程内部调用了run()方法

  1. public void run()
  2. {
  3. sync.innerRun();
  4. }
  5. protected void set(Object obj)
  6. {
  7. sync.innerSet(obj);
  8. }
  9. public Object get()
  10. throws InterruptedException, ExecutionException
  11. {
  12. return sync.innerGet();
  13. }

我们的调用都深入到了sync的方面里面去了。接下来

  1. void innerRun()
  2. {
  3. if(!compareAndSetState(0, 1))
  4. return;
  5. runner = Thread.currentThread();
  6. if(getState() == 1)
  7. {
  8. Object obj;
  9. try
  10. {
  11. obj = callable.call(); //这时候调用我们覆写的call方法了
  12. }
  13. catch(Throwable throwable)
  14. {
  15. setException(throwable);
  16. return;
  17. }
  18. set(obj);//执行完之后将结果放入存起来
  19. } else
  20. {
  21. releaseShared(0);
  22. }
  23. }

如何存,很简单,存入事先准备好的属性中

  1. void innerSet(Object obj)
  2. {
  3. int i;
  4. do
  5. {
  6. i = getState();
  7. if(i == 2)
  8. return;
  9. if(i == 4)
  10. {
  11. releaseShared(0);
  12. return;
  13. }
  14. } while(!compareAndSetState(i, 2));
  15. result = obj;//这么存的
  16. releaseShared(0);
  17. done();
  18. }

如何取

    1. Object innerGet()
    2. throws InterruptedException, ExecutionException
    3. {
    4. acquireSharedInterruptibly(0);
    5. if(getState() == 4)
    6. throw new CancellationException();
    7. if(exception != null)
    8. throw new ExecutionException(exception);
    9. else
    10. return result;

JAVA多线程解惑之多线程返回值的更多相关文章

  1. 测试 多线程 实现 callable 带返回值

    package threadTest; import java.util.ArrayList; import java.util.Date; import java.util.concurrent.C ...

  2. Java的Object.hashCode()的返回值到底是不是对象内存地址?

    关于这个问题,查阅了网上的资料,发现证明过程太繁琐,这里我用了反证法. java.lang.Object.hashCode()的返回值到底是不是对象内存地址? hashCode契约 说到这个问题,大家 ...

  3. Java多线程-新特性-有返回值的线程

    在Java5之前,线程是没有返回值的,常常为了“有”返回值,破费周折,而且代码很不好写.或者干脆绕过这道坎,走别的路了. 现在Java终于有可返回值的任务(也可以叫做线程)了. 可返回值的任务必须实现 ...

  4. Java多线程-Callable的Future返回值的使用

    一般使用线程池执行任务都是调用的execute方法,这个方法定义在Executor接口中: public interface Executor { void execute(Runnable comm ...

  5. Callable+ThreadPoolExecutor实现多线程并发并获得返回值(转)

    出处:https://blog.csdn.net/kity9420/article/details/80740466 前言 经常会遇到一些性能问题,比如调用某个接口,可能要循环调用100次,并且需要拿 ...

  6. Java用FutureTask实现又返回值的线程

    要实现有返回值的多线程,具体代码如下: package thread; import java.util.concurrent.Callable; import java.util.concurren ...

  7. java使用Callable创建又返回值的线程

    并发编程使我们可以将程序分为很多个分离的,相互之间独立的任务,通过使用多线程的机制,将每个任务都会有一个执行线程来单独的驱动,一个线程是 进程中一个单一顺序控制流,一个进程可以拥有多个线程,也就相当于 ...

  8. 慕课网-Java入门第一季-7-3 Java 中无参带返回值方法的使用

    来源:http://www.imooc.com/code/1579 如果方法不包含参数,但有返回值,我们称为无参带返回值的方法. 例如:下面的代码,定义了一个方法名为 calSum ,无参数,但返回值 ...

  9. Java 中带参带返回值方法的使用

    如果方法既包含参数,又带有返回值,我们称为带参带返回值的方法. 例如:下面的代码,定义了一个 show 方法,带有一个参数 name ,方法执行后返回一个 String 类型的结果 调用带参带返回值的 ...

随机推荐

  1. [Irving]SqlServer 拆分函数用法

    drop function dbo.split  删除自定义函数 dbo.split  函数名 函数 dbo.splitcreate function dbo.split(@c varchar(100 ...

  2. 多线程模式之MasterWorker模式

    多线程模式之MasterWorker模式 Master-Worker模式的核心思想是,系统由两类进程协作工作:Master进程和Worker进程.Master进程负责接收和分配任务,Worker进程负 ...

  3. 【CSS】Beginner1:Applying CSS

    CSS(Cascading Style Sheets)   1.Applying CSS Three ways: 1.In-line 2.Internal 3.External   2.In-line ...

  4. Windows Azure -Azure 网站、云服务和虚拟机的对比

    Azure 网站.云服务和虚拟机对比 概述 Azure提供了几种方法来承载网站: Azure网站.云服务和虚拟机.本文帮助您了解选项和为您的Web应用程序做出正确选择. Azure网站是大多数web应 ...

  5. https://lua-toolbox.com/

    Lua Toolbox https://github.com/bungle/lua-resty-session

  6. 10个强大的Apache开源模块

    1.单点登录模块 LemonLDAP LemonLdap可以很棒地实现Apache的SSO功能,并且可以处理超过 20 万的用户请求.LemonLdap支持Java, PHP, .Net, Perl, ...

  7. WPF的MVVM模式

    Model->數據模型View->視圖View-Model->連接數據模型和視圖

  8. (二)如何在.net中使用Redis

    Step1:使用NuGet工具安装Redis C# API,这里有多个API我们可以使用其中一个:

  9. SQLite数据库如何存储和读取二进制数据

    SQLite数据库如何存储和读取二进制数据 1. 存储二进制数据 SQLite提供的绑定二进制参数接口函数为: int sqlite3_bind_blob(sqlite3_stmt*, int, co ...

  10. Rational Rose 2007 &amp;Rational Rose 2003 下载及破解方法和汉化文件下载

    这么好的东西,不拿来出分享,我对不起原作者呀.可是我这里不知道作者是谁,感谢在先了. ed2k://|file|%5BIBM%E8%BD%AF%E4%BB%B6%E7%B3%BB%E5%88%97%5 ...