当前位置: 首页 > news >正文

五、Callable 接口

要和 1.6Runnable 和 Thread 类,几种创建线程方式比较。

创建线程几种方式呢?5 种

  1. 继承 Thread 类
  2. Runnable 接口
  3. Java 5 后,实现 Callable 接口
  4. Java 线程池
  5. ComplatableFuture

5.1 Callable 和 Runnable 区别

有三个区别:

  1. 具体方法不同:一个是run,一个是call
  2. Runnable没有返回值;Callable可以返回执行结果,是个泛型
  3. Callable接口的call()方法允许抛出异常;Runnable的run()方法异常只能在内部消化,不能往上继续抛
FutureTask futureTask = new FutureTask(new Callable());
Thread t = new Thread(FutureTask);
t.start();
Thread t = new Thread(new Runnable());
t.start();

5.2 Callable 创建线程

  • new Thread()是不接收Callable对象的,只能接收Runnable对象,那到底该如何使用Callable呢?

这时候借助一个FutureTask -> RunnableFuture -> Runnable

所以其实根本上,还是因为它实现了RunnableFuture接口,这个接口继承于Runnable接口

所以肯定是可以运行起来了!

new Thread(new FutureTask<>(new CallableFun()), "Callable").start();

class CallableFun implements Callable<Integer> {@Overridepublic Integer call() throws Exception {Thread.sleep(6);System.out.println(Thread.currentThread().getName() + "Callable is running!");return 666;}
}class RunnableFun implements Runnable {@Overridepublic void run() {try {Thread.sleep(10);} catch (InterruptedException e) {throw new RuntimeException(e);}System.out.println(Thread.currentThread().getName() + "Runnable is running!");}
}public class CallableDemo {public static void main(String[] args) {new Thread(new RunnableFun(), "RunableFun").start();new Thread(new FutureTask<>(new CallableFun()), "Callable").start();}
}

  • 那么如何处理返回值呢?

使用Thread进行过去,发现没有那样的方法

这里使用FutureTask.get进行获取返回值,因为嵌在它里面,也可以理解

public class CallableDemo {public static void main(String[] args) throws ExecutionException, InterruptedException {new Thread(new RunnableFun(), "RunableFun").start();FutureTask<Integer> futureTask = new FutureTask<>(new CallableFun());new Thread(futureTask, "CallableFun").start();System.out.println("获取到结果之前");Integer res = futureTask.get();System.out.println("获取到结果之后" + res);}
}

主线程,在 Callable 执行完之后才打印值,就是等他们执行完才得到返回值,get 方法会阻塞主线程。

  • 因为Callable接口有throw可以处理异常,所以需要抛出去
  • 需要将睡眠时间更改一下
class CallableFun implements Callable<Integer> {@Overridepublic Integer call() throws Exception {Thread.sleep(1000);System.out.println(Thread.currentThread().getName() + " Callable is running!");return 666;}
}class RunnableFun implements Runnable {@Overridepublic void run() {try {Thread.sleep(500);} catch (InterruptedException e) {throw new RuntimeException(e);}System.out.println(Thread.currentThread().getName() + " Runnable is running!");}
}

5.3 FutureTask使用

一个FutureTask对象之后就只有一个结果输出,虽然是两个线程

public static void main(String[] args) throws ExecutionException, InterruptedException {FutureTask<Integer> futureTask = new FutureTask<>(new CallableFun());Thread t1 = new Thread(futureTask, "线程 1");Thread t2 = new Thread(futureTask, "线程 2");t1.start();t2.start();System.out.println("获取到结果之前");Integer res = futureTask.get();System.out.println("获取到结果之后" + res);
}

2 个FutureTask对象之后才会有 2 个结果输出

class CallableFun implements Callable<Integer> {@Overridepublic Integer call() throws Exception {Thread.sleep(500);System.out.println(Thread.currentThread().getName() + " Callable is running!");return new Random().nextInt(1000);}
}public static void main(String[] args) throws ExecutionException, InterruptedException {FutureTask<Integer> futureTask1 = new FutureTask<>(new CallableFun());FutureTask<Integer> futureTask2 = new FutureTask<>(new CallableFun());Thread t1 = new Thread(futureTask1, "线程 1");Thread t2 = new Thread(futureTask2, "线程 2");t1.start();t2.start();System.out.println("获取到结果之前");Integer res1 = futureTask1.get();Integer res2 = futureTask2.get();System.out.println("获取到结果之后" + res1);System.out.println("获取到结果之后" + res2);
}

FutureTask的方法

  • get:阻塞主线程
  • isDone:判断异步线程是否执行完毕
  • isCancelled:是否被取消了
  • cancel
    • true:打断子线程执行,并且抛出异常,无返回结果
    • false:不打断子线程执行,并且抛出异常,无返回结果

isDone方法:就是另一种判断是否get完成的方法

class CallableFun implements Callable<Integer> {@Overridepublic Integer call() throws Exception {for (int i = 0; i < 10; i++) {Thread.sleep(500);System.out.println(Thread.currentThread().getName() + " Callable is running!");}return new Random().nextInt(1000);}
}public class CallableDemo {public static void main(String[] args) throws ExecutionException, InterruptedException {FutureTask<Integer> futureTask = new FutureTask<>(new CallableFun());new Thread(futureTask, "线程 1").start();System.out.println();;System.out.println("get 前:" + futureTask.isDone());Integer res = futureTask.get();System.out.println("get 后:" + futureTask.isDone());System.out.println("获取到结果:" + res);}
}

isCancelled、cancel方法

这里Thread.sleep是为了让主线程睡眠,子线程继续执行futureTask任务,来判断是否让子线程继续执行。

package CallableInter;import java.util.Random;
import java.util.concurrent.Callable;
import java.util.concurrent.ExecutionException;
import java.util.concurrent.FutureTask;class CallableFun implements Callable<Integer> {@Overridepublic Integer call() throws Exception {for (int i = 0; i < 10; i++) {Thread.sleep(500);System.out.println(Thread.currentThread().getName() + " Callable is running!");}return new Random().nextInt(1000);}
}public class CallableDemo {public static void main(String[] args) throws ExecutionException, InterruptedException {//        new Thread(new RunnableFun(), "RunableFun").start();FutureTask<Integer> futureTask = new FutureTask<>(new CallableFun());new Thread(futureTask, "线程 1").start();System.out.println("get 前是否 Done:" + futureTask.isDone());System.out.println("get 前取消:" + futureTask.isCancelled());Thread.sleep(3000);futureTask.cancel(false);Integer res = futureTask.get();System.out.println("get 后取消:" + futureTask.isCancelled());System.out.println("get 后是否 Done:" + futureTask.isDone());System.out.println("获取到结果:" + res);}
}

false不打断子线程的执行,会执行结束

http://www.dtcms.com/a/356765.html

相关文章:

  • Spring为什么需要三级缓存
  • 海康相机开发---设备布防(Setup Alarm)
  • 数据库Mysql
  • 微积分 | 积分代换
  • 如何将iPhone日历传输到电脑
  • 内置高压MOS的智能调光方案:AP5126 LED降压恒流驱动芯片
  • 深度拆解判别式推荐大模型RankGPT!生成式精排落地提速94.8%,冷启动效果飙升,还解决了传统推荐3大痛点
  • 评价指标FID/R Precision
  • 《R for Data Science (2e)》免费中文翻译 (第6章) --- scripts and projects
  • 学习游戏制作记录(音频的制作和使用)8.28
  • 算法题打卡力扣第169题:多数元素(easy)
  • 【二叉树(DFS)- LeetCode】124. 二叉树中的最大路径和
  • 考民航安检员证需要具备哪些技能和知识?
  • 卷积神经网络为什么要填充(Padding)
  • Python爬虫实战:研究Pyplot模块,构建IMDb数据采集和分析系统
  • 【Tools】C#文件自动生成UML图
  • Vue3 全面介绍
  • ArcGIS Pro 地图打包与解包
  • STM32CubeMX + HAL 库:基于 I²C 通信的 AHT20 高精度温湿度测量实验
  • 佳易王钓场计时计费系统:全方位赋能钓场智能化管理,软件操作教程
  • Halcon那些事:如何使用差异模型create_variation_model检测印刷品缺陷
  • 日语学习-日语知识点小记-构建基础-JLPT-N3阶段(22):文法+单词第7回4 + 考え方1
  • Java全栈开发工程师面试实录:从基础到微服务的深度技术解析
  • LangChain如何使用通义千问的向量模型
  • 嵌入式学习日志————MPU6050简介
  • 2025年社交广告投放技术解析:应对CPM上涨的核心策略与实战方法
  • 元宇宙与娱乐产业:沉浸式体验与内容生态的重构
  • 前端工程化深度实践:从构建优化到CI/CD的完整解决方案
  • 基于无人机的风电叶片全自动智能巡检:高精度停角估计与细节优先曝光调控技术
  • 【Flask】测试平台开发,产品管理实现添加功能-第五篇