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

看下面例子,我定义了一个类实现了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. codeforce 660D Number of Parallelograms

    题意:询问多少个矩形. 统计横纵坐标差,放进vector中 #include<cstdio> #include<cstring> #include<iostream> ...

  2. UVALive-4287 Proving Equivalences 有向图的强连通分量+缩点

    题意:有n个命题,已知其中的m个推导,要证明n个命题全部等价(等价具有传递性),最少还需要做出几次推导. 思路:由已知的推导可以建一张无向图,则问题变成了最少需要增加几条边能使图变成强连通图.找出所有 ...

  3. Amabri安装前的准备工作

    Ambari这个东东对操作系统的要求还是很高的,首先限制了要使用CentOS,RHEL,并且还是64bit 的,让我的Ubuntu泪奔啊---手头正好有RHEL6.x的盘,于是就用红帽吧.但是红帽是收 ...

  4. ZOJ-3720 Magnet Darts 计算几何,概率

    题目链接:http://acm.zju.edu.cn/onlinejudge/showProblem.do?problemCode=3720 题意:在一个矩形区域投掷飞镖,每个整点有磁性,每个点的磁性 ...

  5. POJ2429 - GCD & LCM Inverse(Miller–Rabin+Pollard's rho)

    题目大意 给定两个数a,b的GCD和LCM,要求你求出a+b最小的a,b 题解 GCD(a,b)=G GCD(a/G,b/G)=1 LCM(a/G,b/G)=a/G*b/G=a*b/G^2=L/G 这 ...

  6. 微软2016校园招聘4月在线笔试 ABC

    题目链接:http://hihocoder.com/contest/mstest2016april1/problems 第一题:输入N,P,W,H,代表有N段文字,每段有ai个字,每行有⌊W/S⌋个字 ...

  7. 坑爹的IE quirk模式【转】

    调试一个页面,ie下面页面css样式很是奇怪,各种失效.找了半天原因不知道怎么回事.最后在调试工具中发现,文档模式为quirk,改成别的(IE 7|8|9)正常. 为什么会自动选择此文档模式呢? 先看 ...

  8. js showModalDialog打开新的页面给原页面传值问题

    a.html中打开一个新页面b.html,b.html页面给a.html中的input传一个值并将value赋给input框. a.html: <html>  <head>   ...

  9. Windows Server 2012 R2中的网络诊断命令

    Get-NetAdapter Get-NetIPAddress Get-NetIPConfiguration(GIP) TNC :Pinging Servers and Trace Route tnc ...

  10. Test-NetConnection

    有了PowerShell 4.0,排除网络故障的日子就会轻松很多.Test-NetConnection cmdlet将多个测试整合到了一个简单的有几个参数的命令当中.例如,命令Test-NetConn ...