runnable和callable区别和底层原理
确实,`Runnable` 可以直接通过 `Thread` 类来运行,而 `Callable` 不能直接用于创建和运行线程。`Callable` 和 `Runnable` 都是 Java 中用于定义异步任务的接口,但它们的用法和目的有所不同。
### Runnable 和 Thread
`Runnable` 是接口,它不返回结果也不抛出检查异常。通常,你可以直接将 `Runnable` 传递给 `Thread` 对象来创建并启动线程。
```java
Runnable runnableTask = () -> {
// 任务逻辑
System.out.println("Runnable task is running");
};
Thread thread = new Thread(runnableTask);
thread.start();
```
### Callable 和 FutureTask
`Callable` 是一个有返回值的任务接口,提供更丰富的功能(如抛出受检异常和返回结果)。但它不能直接用于创建线程。为了执行 `Callable`,你需要使用 `FutureTask` 或其他与执行器相关的机制。
- **FutureTask** 是一种实现了 `Runnable` 和 `Future` 接口的类,允许你将 `Callable` 转换成一个 `Runnable` 对象,以便它可以被线程执行,并提供与 `Future` 相同的特性来处理结果。
```java
import java.util.concurrent.Callable;
import java.util.concurrent.FutureTask/;
public class CallableExample {
public static void main(String[] args) throws Exception {
Callable<String> callableTask = () -> {
// 模拟长时间任务
Thread.sleep(1000);
return "Task's result";
};
// 使用 FutureTask 包装 Callable
FutureTask<String> futureTask = new FutureTask<>(callableTask);
// 使用 Thread 执行 FutureTask
Thread thread = new Thread(futureTask);
thread.start();
// 获取任务执行结果
String result = futureTask.get();
System.out.println("Result from Callable: " + result);
}
}
```
### 关键点
- **Runnable 和 Thread**:`Runnable` 可直接通过 `Thread` 使用来执行任务。适用于不需要返回值或处理异常的简单任务。
- **Callable 和 FutureTask**:`Callable` 需要通过 `FutureTask` 或 `ExecutorService` 来启动执行并获取结果。适用于需要结果或处理异常的复杂任务。
通过 `FutureTask`,`Callable` 可以集成到传统的线程启动模式中。使用 `ExecutorService` 可能是更常见的方法,特别是在管理复杂应用程序的线程资源和并发执行时。
可以说,`Callable` 和 `Runnable` 在 Java 的并发编程中是从不同的角度设计出来的,但它们在实现任务执行时确实有一些重叠。在底层,`Callable` 任务最终通常是通过某种方式转换成 `Runnable` 来运行的。这是因为 Java 的 `Thread`只接受 `Runnable` 接口,而不直接支持 `Callable`。
### 理解 `Callable` 和 `Runnable` 的关系
1. **Runnable**:
- 是一个标准的接口,用于定义没有返回值和不抛出检查异常的任务。
- 是一个简单的结构,可以直接被 `Thread` 执行,如 `new Thread(myRunnable).start();`
2. **Callable**:
- 是 Java 5 引入的,用于定义返回结果、可以抛出检查异常的任务。
- 为了运行 `Callable`,需要转换成 `Runnable`,这通常通过 `FutureTask` 或通过 `ExecutorService` 实现。
3. **FutureTask**:
- `FutureTask` 是一类实现了 `Runnable` 的适配器,它包装了 `Callable`,以便它能够被线程执行。
- `FutureTask` 内部通过实现 `Runnable` 的 `run` 方法来执行 `Callable` 并管理其生命周期。
```java
Callable<String> callableTask = () -> "Callable Result";
FutureTask<String> futureTask = new FutureTask<>(callableTask);
Thread thread = new Thread(futureTask);
thread.start();
String result = futureTask.get();
```
在这个例子中,你可以看到 `FutureTask` 是如何充当中介,将 `Callable` 转换为 `Runnable` 以便通过 `Thread` 执行的。
### `ExecutorService` 的实现
- 当你使用 `ExecutorService.submit(Callable)` 时,执行器可能也是通过类似的方式将 `Callable` 包装成 `FutureTask`,然后运行它。
- `ExecutorService` 基本上使用线程池去管理线程的创建和执行,并为任务(包括 `Runnable` 和 `Callable`)提供统一的调度与管理框架。
### 小结
虽然 `Callable` 不直接实现 `Runnable`,但是它们被设计为构建在统一的执行机制上,因此在实际执行过程中常常通过某种适配器(如 `FutureTask`)来间接地使用 `Runnable` 的执行模型。这种设计允许 `Callable` 提供额外的功能优势,同时保持与现有线程执行架构的兼容性。
相关文章:
runnable和callable区别和底层原理
确实,Runnable 可以直接通过 Thread 类来运行,而 Callable 不能直接用于创建和运行线程。Callable 和 Runnable 都是 Java 中用于定义异步任务的接口,但它们的用法和目的有所不同。 ### Runnable 和 Thread Runnable 是接口,它不返…...
Springboot 整合 Java DL4J 打造自然语言处理之语音识别系统
🧑 博主简介:CSDN博客专家,历代文学网(PC端可以访问:https://literature.sinhy.com/#/literature?__c1000,移动端可微信小程序搜索“历代文学”)总架构师,15年工作经验,…...
虚幻引擎5(UE5)学习教程
虚幻引擎5(UE5)学习教程 引言 虚幻引擎5(Unreal Engine 5,简称UE5)是Epic Games开发的一款强大的游戏引擎,广泛应用于游戏开发、影视制作、建筑可视化等多个领域。UE5引入了许多先进的技术,如…...
从0开始深度学习(26)——汇聚层/池化层
池化层通过减少特征图的尺寸来降低计算量和参数数量,同时增加模型的平移不变性和鲁棒性。汇聚层的主要优点之一是减轻卷积层对位置的过度敏感。 1 最大汇聚层、平均汇聚层 汇聚层和卷积核一样,是在输入图片上进行滑动计算,但是不同于卷积层的…...
兼职发薪系统:高效、便捷的劳务发薪解决方案
在快速发展的数字化时代,企业对于高效、便捷的薪酬发放和管理解决方案的需求日益增长。特别是对于兼职人员众多的企业,如何实现快速、准确的发薪,同时确保员工信息的安全与保密,成为了一个亟待解决的问题。今天,我们将…...
MySQL数据库单表查询习题
目录 数据内容介绍习题题目答案 数据内容介绍 数据库中有两个表 内容如下: 习题 题目 查询出部门编号为D2019060011的所有员工所有财务总监的姓名、编号和部门编号。找出奖金高于工资的员工。找出奖金高于工资40%的员工。找出部门编号为D2019090011中所有…...
多模态PaliGemma——Google推出的基于SigLIP和Gemma的视觉语言模型
前言 本文怎么来的呢?其实很简单,源于上一篇文章《π0——用于通用机器人控制的流匹配VLA模型:一套框架控制7种机械臂(改造了PaliGemma和ACT的3B模型)》中的π0用到了PaliGemma 故本文便来解读下这个PaliGemma 第一部分 PaliGemma 1.1 Pal…...
电路原理:电阻桥。
电路的基础是电阻电路。电阻电路有两种基本接线方法(串连和并连,二者有不同的解算与用法:串连分压、并连分流)。电阻电路就是使用基本接线方法的组合方案,其解算方法主要内容是判断好整体布局以及各个局部的串并连关系…...
实践出真知:MVEL表达式中for循环的坑
目录标题 背景MVEL脚本(有问题的)MVEL脚本(正确的)结论分析 背景 需要从一个URL的拼接参数中解析出id的值并输出 比如: 存在URLhttps://xxxxxxxxxx?id999999&type123&name345 然后需要输出id999999 MVEL脚本(有问题的) 入参:parseThisUrlhttp…...
Flutter运行App时出现“Running Gradle task ‘assembleDebug“问题解决
在参考了众多解决办法中最有用并且最快的方法 Gradle Distributions 在这个地方下载对应你这个文件中的gradle版本 然后将 最后一行本来不是这样的,我们把下载好的zip包保存到本地,然后用这个代替网址,最后成功运行...
基于SSM(Spring + Spring MVC + MyBatis)框架的咖啡馆管理系统
基于SSM(Spring Spring MVC MyBatis)框架的咖啡馆管理系统是一个综合性的Web应用程序,用于管理和优化咖啡馆的运营。下面我将提供一个详细的案例程序概述,包括主要的功能模块和技术栈介绍。 项目概述 功能需求 用户管理&…...
【SpringBoot】18 上传文件到数据库(Thymeleaf + MySQL)
Git仓库 https://gitee.com/Lin_DH/system 介绍 使用 Thymeleaf 写的页面,将(txt、jpg、png)格式文件上传到 MySQL 数据库中。 依赖 pom.xml <!-- https://mvnrepository.com/artifact/com.mysql/mysql-connector-j --><depende…...
计算机体系结构之系统吞吐量(三)
前面章节《计算机体系结构之多级缓存、缓存miss及缓存hit(二)》讲了关于系统多级缓存的相关内容,其中提及了系统吞吐量一词。在此章将对其进行讲解。 系统吞吐量是计算机体系结构的一个重要指标,其衡量的是系统在单位时间内处理工…...
高级 HarmonyOS主题课—— 帮助快速构建各种文本识别应用的课后习题
天地不仁,以万物为刍狗; 圣人不仁,以百姓为刍狗。 天地之间,其犹橐龠乎? 虚而不屈,动而俞出。 多闻数穷,不若守于中。 本文内容主要来自 <HarmonyOS主题课>帮助快速构建各种文本识别应用 …...
windows C#-异常和异常处理概述
C# 语言的异常处理功能有助于处理在程序运行期间发生的任何意外或异常情况。 异常处理功能使用 try、catch 和 finally 关键字来尝试执行可能失败的操作、在你确定合理的情况下处理故障,以及在事后清除资源。 公共语言运行时 (CLR)、.NET/第三方库或应用程序代码都可…...
每日一题——第一百二十四题
题目:进制转换 #pragma once#include<stdio.h> #include<ctype.h> #include<string.h>/// <summary> /// //将字符串表示的任意进制数转为十进制 /// </summary> /// <param name"str">字符串</param> /// &l…...
在 CentOS 7 上设置 OpenResty 开机启动
在 CentOS 7 上设置 OpenResty 开机启动,可以按照以下步骤进行操作: 创建 Systemd 服务文件: 首先,您需要为 OpenResty 创建一个 Systemd 服务文件。使用文本编辑器(如 vi 或 nano)创建一个新的服务文件。 …...
势不可挡 创新引领 | 生信科技SOLIDWORKS 2025新品发布会·苏州站精彩回顾
2024年11月01日,由生信科技举办的SOLIDWORKS 2025新产品发布会在江苏苏州圆满落幕。现场邀请到制造业的专家学者们一同感受SOLIDWORKS 2025最新功能,探索制造业数字化转型之路。 在苏州站活动开场,达索系统专业客户事业部华东区渠道经理马腾飞…...
数仓之全量表、增量表、快照表、切片表、拉链表的基本概念
文章摘自:数仓之全量表、增量表、快照表、切片表、拉链表-腾讯云开发者社区-腾讯云 一、全量表 记录每天所有最新状态的数据,有无变化都要上报,每次往全量表里面写数据都会覆盖之前的数据 缺点:不能记录数据的历史变化ÿ…...
【富集分析GSEA】如何理解富集分析以及应用
如何理解富集分析 富集分析不同的方式 富集分析 不同的方式 直接使用疾病特征进行富集分析(不翻转上调和下调的基因) 目的:如果你的目标是了解疾病状态的生物学特征和功能路径,那么应该直接使用疾病特征(包含疾病状态…...
基于ARM核心板的T-BOX系统设计:从硬件选型到软件实现
1. 项目概述与核心价值最近几年,车联网的概念已经从实验室和展会,实实在在地走进了我们的日常生活。作为一名在嵌入式领域摸爬滚打了十几年的工程师,我亲眼见证了从简单的GPS定位模块,到如今功能高度集成的车载T-BOX(T…...
保姆级教程:用STM32F103C8T6和MAX485芯片实现稳定的一主多从RS485通讯(附完整代码)
STM32F103C8T6与MAX485构建工业级RS485总线系统实战指南 在工业自动化领域,稳定可靠的通信系统如同神经脉络般重要。想象一下,当您需要在一个大型温室中部署数十个温湿度传感器,或者在一个工厂车间监控多台设备的运行状态时,点对点…...
保姆级教程:把Windows系统装进固态U盘,用云固件打造随身移动办公神器
随身Windows系统:用固态U盘打造移动办公终极解决方案 咖啡馆的午后阳光斜照在键盘上,你从包里掏出一个名片大小的设备,插入陌生电脑的USB接口。30秒后,熟悉的桌面环境、未写完的文档、收藏夹里的书签全部跃然屏上——这不是科幻场…...
量子态重构技术QSDC:动态电路与机器学习结合
1. 量子态重构的技术挑战与QSDC框架概述 量子计算领域长期面临一个基础性难题:如何在电路运行过程中获取量子态的"快照"而不破坏其量子特性?传统量子态层析(QST)需要制备大量相同量子态副本进行测量,不仅效率…...
[STM32U3] 【STM32U385RG 测评】PWM调节屏幕亮度
在评测计划中有使用pwm来实现调节屏幕亮度,因此本篇为如何使用HMI实现对屏的亮度调节。实现原理为,使用TouchGFX Designer添加一个滑动控件,通过滑动来修改pwm的占空比,实现ST7789的BLK的电压实现。 本次工程在上一篇试用的基础上…...
学习规划需要定期调整吗?
在当今竞争激烈的教育环境中,学习规划对于学生的成长和发展起着至关重要的作用。作为一名在学习规划领域深耕十年的专家,我见证了无数学生在学习规划的指引下取得优异成绩,也看到了一些学生因为规划不合理而走了不少弯路。那么,学…...
告别插线!用ESP32的OTA Web Updater实现无线烧录,保姆级避坑指南
ESP32无线固件更新全攻略:从零构建OTA Web Updater系统 引言:为什么需要无线更新? 想象一下,你精心设计的智能温室控制系统已经安装在屋顶的密闭箱体中,突然发现需要修复一个关键的温度传感器逻辑错误。传统方式需要…...
保姆级教程:Win10/Win11下彻底解决原神启动器Qt插件初始化失败(附环境变量排查与恢复指南)
深度解析Windows环境下Qt插件初始化失败的终极解决方案 当你在Windows 10或11系统上双击原神启动器,却看到"no Qt platform plugin could be initialized"的错误提示时,那种挫败感不言而喻。这个问题看似简单,实则涉及系统环境变量…...
实测测评|零注册AI PDF翻译工具:保留排版\+OCR无损翻译,替代DeepL/谷歌翻译
在日常开发、学术科研、外文文献研读场景中,PDF翻译一直是高频刚需痛点。市面上主流的翻译工具普遍存在排版错乱、OCR收费、文件大小受限、强制登录注册等问题,尤其是学术论文、带表格/公式的技术手册、扫描版外文资料,翻译后的可用性极差。 …...
别等 SQL 跑慢了再靠猜,NineData 怎样把问题一步步看清
SQL 变慢最怕的不是慢本身,而是大家一开始只能靠猜。明明语法没报错,业务一上线却卡住了;多了一个函数、少了一个索引,或者关联顺序不合适,就可能让数据库白白扫描一大堆数据。在很多团队里,这类问题最早往…...
