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

看下面例子,我定义了一个类实现了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. PLS-00306:错误解决思路 - OracleHelper 执行Oracle函数的坑

    如果你是像我一样初次使用Net+Oracle的结合,我想你会跟我一样,有很大的概率碰到这个问题 ==================================================== ...

  2. js定义函数的几种结构形式

    1.函数声明 function name(参数) { 执行的代码 } 2.函数表达式 也称匿名函数 var x = function (参数) { 执行的代码 }; //第二种方式按照完整的语法需要在 ...

  3. 【转】RDO、SAD、SATD、λ

    SAD(Sum of Absolute Difference)=SAE(Sum of Absolute Error)即绝对误差和 SATD(Sum of Absolute Transformed Di ...

  4. Linux下简单的socket通信实例

    Linux下简单的socket通信实例 If you spend too much time thinking about a thing, you’ll never get it done. —Br ...

  5. Sublime Text2使用心得

    代码编辑器或者文本编辑器,对于程序员来说,就像剑与战士一样,谁都想拥有一把可以随心驾驭且锋利无比的宝剑,而每一位程序员,同样会去追求最适合自己的强大.灵活的编辑器,相信你和我一样,都不会例外. 我用过 ...

  6. (1)I2c的简介和特性

    I2C我是想全面深入的从嵌入式软件工程师的角度做个理解,刚刚还申请了一个专栏,这个好好写.         学习技术从外文文档看起-- 要全面了解I2C,可以从<I2C-bus specific ...

  7. Azure VM对远程桌面登录的支持-示例

    我们在开发Windows Azure的应用程序,虽然在大部分的情况下都可以使用Azure Emulator模拟器来模拟在云端计算节点(Azure VM)的执行结果,但是并不能100%模拟真正在Azur ...

  8. opengl (1) 基本API的熟悉

    代码从此处下载 1 运行如下代码,可以看到如下效果,我们利用opengl画出一个三角形. void renderScene(void) { /* glClear清除缓冲区 */ glClear(GL_ ...

  9. 第一章 Windows NT System Components

    Page 3. The focus(焦点) of this book is Windows NT file system and the interaction(交互) of the file sys ...

  10. 【腾讯优测干货分享】如何降低App的待机内存(五)——优化dex相关内存及本章总结

    本文来自于腾讯优测公众号(wxutest),未经作者同意,请勿转载,原文地址:http://mp.weixin.qq.com/s/01Abwe0p1h3WLh28Tzg_Dw 1.5案例:优化dex相 ...