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

看下面例子,我定义了一个类实现了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. Codevs 1222 信与信封问题 二分图匹配,匈牙利算法

    题目: http://codevs.cn/problem/1222/ 1222 信与信封问题   时间限制: 1 s   空间限制: 128000 KB   题目等级 : 钻石 Diamond 题解 ...

  2. Esper系列(十二)Variables and Constants

    功能:变量和常量的定义及应用. EPL配置创建 1   "); 3  // 创建 orderBean 事件类型变量 bean 4  epAdmin.createEPL("creat ...

  3. HDOJ-ACM1006(JAVA)

    题意:输入一个角度degree,求出一天中时针分针秒针之间的角度大于这个角度degree的时间占一天总时间的比例 因此输入是0-120度, 输出比例,保留三位小数,-1为终止 暂时没想出来如何做这道题 ...

  4. linux bin文件制作

    一 Linux安装文件 Linux常见的安装为tar,zip,gz,rpm,deb,bin等.我们可以简单的分为三类, 第一:打包或压缩文件tar,zip,gz等,一般解压后即可,或者解压后运行sh文 ...

  5. 源码生成deb包

    方法一 源码包要求是使用 automake 进行编译管理的. 安装路径不能指定为 /usr/local 下的目录,否则生成 deb 包期间报错. 制作的工具是 dh-make ,如果没有安装,要先安装 ...

  6. centos6 kvm网卡桥接

    以前用VMware,我的上司说,你既然都用CentOS的桌面,那就研究一下KVM. 好吧,上司做运维好几年了,就听了他的,装了一个KVM. KVM的网络默认是NAT,不方便,就学习BRIDGE!!! ...

  7. hdu5601-N*M bulbs(黑白棋盘染色)

    一个矩形,一个人从左上角走到右下角,每走过一个位置把0变成1,1变成0. 求有没有可能他离开之后所有的数都是0 假设这个矩形是一个棋盘,黑白相间. 这样会发现从一个颜色走到相同颜色可以对棋盘不产生任何 ...

  8. 解决IE下iframe默认有白色背景的bug

    又是一个IE莫名其妙的bug:做了一个弹出层,层里面是一张透明图片,IE下的iframe默认会有一个白色的背景,所以当iframe的外围背景并不是白色的时候,这个iframe就会显得非常的“与众不同” ...

  9. ActiveMQ内存设置和流控

    启动脚本设置jvm的内存 if "%ACTIVEMQ_OPTS%" == "" set ACTIVEMQ_OPTS=-Xms1G-Xmx1G -Djava.ut ...

  10. JAVA Serialization 序列化

    最近在做Android 项目时用到了WebView,可悲的是,在html上有无数用户的操作,而这些操作被JS返回给了Android的内存中,当深层的Activity开启时,之前的Activity很可能 ...