Java多线程Thread , Future , Callable , FutureTask的使用
Thread创建一个线程
在想要使用多线程的技术来进行相应的操作的时候: 可以含有以下的方向来进行设计思考! 在创建一个线程的时候通常是创建一个Thread的。 但是这个Thread,是含有多种方式来进行创建操作的。 例如: 直接使用空参数的构造器进行创建操作:
System.out.println("start:" + System.currentTimeMillis()); Thread thread = new Thread(); thread.start(); System.out.println("end:" + System.currentTimeMillis());
这样执行是没有什么效果的。这种情况就是你启动了一个线程,但是这个线程是没有任何事情要干的。
时间是过的飞起的。
但是还有一种方法也是可以进行线程的启动操作的。
使用含有参数方法的构造器进行创建线程:
System.out.println(Thread.currentThread() + "----start:" + System.currentTimeMillis()); Thread thread = new Thread(new Runnable() { @Override public void run() { try { Thread.sleep(1000); } catch (InterruptedException e) { e.printStackTrace(); } System.out.println("我正在执行一个任务,不要打扰我!"); System.out.println(Thread.currentThread()); } }); thread.start(); System.out.println(Thread.currentThread() + "----end:" + System.currentTimeMillis());
这种是开启一个线程让其来进行线程任务的执行操作的。
但是你有没有发现,这种方式有一点缺点的。
这个开启的线程是相当于我要让其进行一些子任务的执行操作的。
Thread进行异步处理
类似于我是老师,我想让学习委员去办公室帮我拿一些书来的。此时学习委员已经出发了,但是我想起来我好像是带了书的。我想要让学习委员不要去办公室拿书了。
若我直接采用这种方式来来进行执行操作的话,是无法满足我的需求的。只要是我叫他去拿书了,她就会喊不回来的。
那是否能对这种方式进行相应的改造操作呢???
java的开发者是给了我们一种实现的方式的。
使用Future的接口满足需求。
相应的解释:
A Future 表示异步计算的结果。提供了用于检查计算是否完成、等待计算完成以及检索计算结果的方法。只有在计算完成后才能使用方法 get 检索结果,必要时会阻止,直到它准备就绪。取消是通过该 cancel 方法执行的。提供了其他方法来确定任务是正常完成还是已取消。计算完成后,无法取消计算。如果为了可取消性而使用 a Future ,但不提供可用的结果,则可以声明表单 Future<?> 的类型,并作为基础任务的结果返回 null 。
public interface Future<V> { /** * Attempts to cancel execution of this task. This attempt will * fail if the task has already completed, has already been cancelled, * or could not be cancelled for some other reason. If successful, * and this task has not started when {@code cancel} is called, * this task should never run. If the task has already started, * then the {@code mayInterruptIfRunning} parameter determines * whether the thread executing this task should be interrupted in * an attempt to stop the task. * * <p>After this method returns, subsequent calls to {@link #isDone} will * always return {@code true}. Subsequent calls to {@link #isCancelled} * will always return {@code true} if this method returned {@code true}. * * @param mayInterruptIfRunning {@code true} if the thread executing this * task should be interrupted; otherwise, in-progress tasks are allowed * to complete * @return {@code false} if the task could not be cancelled, * typically because it has already completed normally; * {@code true} otherwise */ boolean cancel(boolean mayInterruptIfRunning); /** * Returns {@code true} if this task was cancelled before it completed * normally. * * @return {@code true} if this task was cancelled before it completed */ boolean isCancelled(); /** * Returns {@code true} if this task completed. * * Completion may be due to normal termination, an exception, or * cancellation -- in all of these cases, this method will return * {@code true}. * * @return {@code true} if this task completed */ boolean isDone(); /** * Waits if necessary for the computation to complete, and then * retrieves its result. * * @return the computed result * @throws CancellationException if the computation was cancelled * @throws ExecutionException if the computation threw an * exception * @throws InterruptedException if the current thread was interrupted * while waiting */ V get() throws InterruptedException, ExecutionException; /** * Waits if necessary for at most the given time for the computation * to complete, and then retrieves its result, if available. * * @param timeout the maximum time to wait * @param unit the time unit of the timeout argument * @return the computed result * @throws CancellationException if the computation was cancelled * @throws ExecutionException if the computation threw an * exception * @throws InterruptedException if the current thread was interrupted * while waiting * @throws TimeoutException if the wait timed out */ V get(long timeout, TimeUnit unit) throws InterruptedException, ExecutionException, TimeoutException; }
通过这个接口就可以实现相对应的功能操作了。
给予了一个接口想要将其进行实现操作就得要相关的实现类。
登场的就是FutureTask的类进行操作的。
public FutureTask(Runnable runnable, V result) { this.callable = Executors.callable(runnable, result); this.state = NEW; // ensure visibility of callable }
执行一下相关的代码:
FutureTask<String> stringFutureTask = new FutureTask<String>(new Runnable() { @Override public void run() { try { Thread.sleep(3000); } catch (InterruptedException e) { e.printStackTrace(); } System.out.println("我正在执行一个任务,不要打扰我!"); System.out.println(Thread.currentThread()); } } , null); Thread futureThread = new Thread(stringFutureTask); futureThread.start(); // 是可以满足对一个进行任务的取消工作的 // stringFutureTask.cancel(true); System.out.println(stringFutureTask.isCancelled()); Thread futureThread1 = new Thread(stringFutureTask); futureThread1.start();
第二个 futureThread1 是不会执行的。
解释原因:
在Java中,FutureTask
只能执行一次。当您尝试第二次执行相同的FutureTask
实例时,它不会执行任何操作。这是因为FutureTask
的状态在第一次执行后会改变,它不会重新开始。如果第一个线程(futureThread
)已经开始执行stringFutureTask
,那么第二个线程(futureThread1
)尝试开始相同的stringFutureTask
时,将不会有任何效果,因为FutureTask
的状态已经不是初始状态了。
这里是一些关键点:
FutureTask
在执行后会进入完成状态。如果尝试再次执行一个已经完成的
FutureTask
,它不会重新执行。如果需要重新执行任务,您需要创建一个新的
FutureTask
实例。
上面相关的例子,已经将一个老师叫学习委员去办公室拿书的例子叫停了。也就是说现在是实现了异步处理的操作。
带有参数的返回线程处理
我现在还有一个需求,是干嘛的呢???我不是喊学习委员去办公室拿书了吗??我想要那个他拿到的书给我手上。这个功能是怎么实现的呢?
也就是含有返回值的相关的例子!!!
JUC给了一个方法,是可以直接将一个Callable的接口传过来的
public FutureTask(Callable<V> callable) { if (callable == null) throw new NullPointerException(); this.callable = callable; this.state = NEW; // ensure visibility of callable }
使用这个接口作为参数传给的FutureTask进行任务的初始化操作。
我们知道Callable接口是有返回值的。那么就可以类似于得到相应的那一本书了!!!
操作实现:
FutureTask<String> stringFutureTask = new FutureTask<String>(new Callable<String>() { @Override public String call() throws Exception { try { Thread.sleep(3000); } catch (InterruptedException e) { e.printStackTrace(); } System.out.println("我正在执行一个任务,不要打扰我!"); return "老师,我拿到书了"; } }); Thread thread = new Thread(stringFutureTask); thread.start(); System.out.println(stringFutureTask.get());
将得到的东西进行返回操作。这样的话就可以实现三种基本的功能操作了。
到此这篇关于Java多线程Thread , Future , Callable , FutureTask的使用的文章就介绍到这了,更多相关Java Thread , Future , Callable , FutureTask内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!
- Java中关于ThreadLocal的隐式引用详解
- Java中的Random和ThreadLocalRandom详细解析
- Java中的ThreadLocal源码及弱引用解析
- Java中ThreadLocal共享变量的使用
- Java中的ScheduledThreadPoolExecutor定时任务详解
- Java中的FutureTask实现代码实例
- Java中的FutureTask源码解析
- 一文搞懂Runnable、Callable、Future、FutureTask及应用
- Java并发编程中的Callable、Future和FutureTask详解
- Java中Future和FutureTask的示例详解及使用
- Java多线程 Callable、Future 和FutureTask
相关文章
JDK安装与配置超级详细教程(包含二个或多个JDK的同时安装)
这篇文章主要给大家介绍了关于JDK安装与配置(包含二个或多个JDK的同时安装)的相关资料,对于Java学习者来说,一台电脑拿到手肯定要配置JDK,但是对于新手来说还是容易出错,需要的朋友可以参考下2023-10-10Java Http请求方式之RestTemplate常用方法详解
这篇文章主要为大家介绍了Java Http请求方式之RestTemplate常用方法详解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪2022-09-09Spring中的注解@Autowired实现过程全解(@Autowired 背后的故事)
这篇文章主要介绍了Spring中的注解@Autowired实现过程全解,给大家聊聊@Autowired 背后的故事及实现原理,需要的朋友可以参考下2021-07-07Mybatis-plus如何查询返回对象内有List<String>属性
在使用Mybatis-Plus进行开发时,我们经常会遇到需要处理一对多关系映射的情况,例如,查询用户数据时,可能需要同时获取该用户管理的所有小区名称列表,这要求我们在返回的实体类中包含一个List<String>属性,用于存放小区名称,实现这一功能2024-10-10
最新评论