JAVA多线程解惑之多线程返回值

2024-04-14 03:32

本文主要是介绍JAVA多线程解惑之多线程返回值,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

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

 

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

 

public class MyCallable implements Callable<Object> {@Overridepublic Object call() throws Exception {int i=10;Thread.sleep(10000);return i;}}

 

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

 

 

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

 

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

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

 

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

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

new Thread(task).start();

 看看这个task是什么的东西

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

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

public interface Future
{public abstract boolean cancel(boolean flag);public abstract boolean isCancelled();public abstract boolean isDone();public abstract Object get()throws InterruptedException, ExecutionException;public abstract Object get(long l, TimeUnit timeunit)throws InterruptedException, ExecutionException, TimeoutException;
}

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

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

    public void run(){sync.innerRun();}protected void set(Object obj){sync.innerSet(obj);}public Object get()throws InterruptedException, ExecutionException{return sync.innerGet();}

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

       void innerRun(){if(!compareAndSetState(0, 1))return;runner = Thread.currentThread();if(getState() == 1){Object obj;try{obj = callable.call(); //这时候调用我们覆写的call方法了}catch(Throwable throwable){setException(throwable);return;}set(obj);//执行完之后将结果放入存起来} else{releaseShared(0);}}

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

void innerSet(Object obj){int i;do{i = getState();if(i == 2)return;if(i == 4){releaseShared(0);return;}} while(!compareAndSetState(i, 2));result = obj;//这么存的releaseShared(0);done();}

 如何取

        Object innerGet()throws InterruptedException, ExecutionException{acquireSharedInterruptibly(0);if(getState() == 4)throw new CancellationException();if(exception != null)throw new ExecutionException(exception);elsereturn result;

 

 

 

 

 

这篇关于JAVA多线程解惑之多线程返回值的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/901975

相关文章

springboot集成easypoi导出word换行处理过程

《springboot集成easypoi导出word换行处理过程》SpringBoot集成Easypoi导出Word时,换行符n失效显示为空格,解决方法包括生成段落或替换模板中n为回车,同时需确... 目录项目场景问题描述解决方案第一种:生成段落的方式第二种:替换模板的情况,换行符替换成回车总结项目场景s

SpringBoot集成redisson实现延时队列教程

《SpringBoot集成redisson实现延时队列教程》文章介绍了使用Redisson实现延迟队列的完整步骤,包括依赖导入、Redis配置、工具类封装、业务枚举定义、执行器实现、Bean创建、消费... 目录1、先给项目导入Redisson依赖2、配置redis3、创建 RedissonConfig 配

SpringBoot中@Value注入静态变量方式

《SpringBoot中@Value注入静态变量方式》SpringBoot中静态变量无法直接用@Value注入,需通过setter方法,@Value(${})从属性文件获取值,@Value(#{})用... 目录项目场景解决方案注解说明1、@Value("${}")使用示例2、@Value("#{}"php

SpringBoot分段处理List集合多线程批量插入数据方式

《SpringBoot分段处理List集合多线程批量插入数据方式》文章介绍如何处理大数据量List批量插入数据库的优化方案:通过拆分List并分配独立线程处理,结合Spring线程池与异步方法提升效率... 目录项目场景解决方案1.实体类2.Mapper3.spring容器注入线程池bejsan对象4.创建

线上Java OOM问题定位与解决方案超详细解析

《线上JavaOOM问题定位与解决方案超详细解析》OOM是JVM抛出的错误,表示内存分配失败,:本文主要介绍线上JavaOOM问题定位与解决方案的相关资料,文中通过代码介绍的非常详细,需要的朋... 目录一、OOM问题核心认知1.1 OOM定义与技术定位1.2 OOM常见类型及技术特征二、OOM问题定位工具

基于 Cursor 开发 Spring Boot 项目详细攻略

《基于Cursor开发SpringBoot项目详细攻略》Cursor是集成GPT4、Claude3.5等LLM的VSCode类AI编程工具,支持SpringBoot项目开发全流程,涵盖环境配... 目录cursor是什么?基于 Cursor 开发 Spring Boot 项目完整指南1. 环境准备2. 创建

Spring Security简介、使用与最佳实践

《SpringSecurity简介、使用与最佳实践》SpringSecurity是一个能够为基于Spring的企业应用系统提供声明式的安全访问控制解决方案的安全框架,本文给大家介绍SpringSec... 目录一、如何理解 Spring Security?—— 核心思想二、如何在 Java 项目中使用?——

SpringBoot+RustFS 实现文件切片极速上传的实例代码

《SpringBoot+RustFS实现文件切片极速上传的实例代码》本文介绍利用SpringBoot和RustFS构建高性能文件切片上传系统,实现大文件秒传、断点续传和分片上传等功能,具有一定的参考... 目录一、为什么选择 RustFS + SpringBoot?二、环境准备与部署2.1 安装 RustF

springboot中使用okhttp3的小结

《springboot中使用okhttp3的小结》OkHttp3是一个JavaHTTP客户端,可以处理各种请求类型,比如GET、POST、PUT等,并且支持高效的HTTP连接池、请求和响应缓存、以及异... 在 Spring Boot 项目中使用 OkHttp3 进行 HTTP 请求是一个高效且流行的方式。

java.sql.SQLTransientConnectionException连接超时异常原因及解决方案

《java.sql.SQLTransientConnectionException连接超时异常原因及解决方案》:本文主要介绍java.sql.SQLTransientConnectionExcep... 目录一、引言二、异常信息分析三、可能的原因3.1 连接池配置不合理3.2 数据库负载过高3.3 连接泄漏