rust语言tokio库spawn, blocking_spawn等的使用
目录
- tokio的spawn以及spawn_blocking的使用
- tokio::task::spawn方法解析
- tokio::task::spawn_blocking()方法解析
时间会遗忘一切
最后更新时间2024.04.29
tokio版本:
tokio的spawn以及spawn_blocking的使用
tokio::task::spawn方法解析
tokio的实现原理以及源码解析请移步我的另一篇博客:
我们举一个实际的例子来说明tokio::spawn的使用。我们创建一个tokio::main,指定工作线程数量为2,方便大家理解,如果不指定,则会与CPU数量相同。因为在这个例子中,我们一共有两个异步sleep,所以创建两个工作线程方便大家理解。
use tokio;#[tokio::main(flavor = "multi_thread", worker_threads = 2)]
async fn main() {let handle_1 = tokio::task::spawn(async {std::thread::sleep(std::time::Duration::from_secs(10));println!("sleeping 10s");});let handle_2 = tokio::task::spawn(async {std::thread::sleep(std::time::Duration::from_secs(1));println!("second spawn!");});tokio::join!(handle_1, handle_2);println!("hello world!");
}
tokio::spawn方法的返回值是一个handle,如果不调用tokio::join!方法,tokio是不会将这两个handle放入工作线程中去运行的。当我们调用了tokio::join!后,相当于同时调用了handl_1.await和handle_2.await,main函数主线程会阻塞等待这两个handle执行完成。所以最终的输出结果是这样的:
# 等1s后打印
second spawn!
# 打印second spawn后,等9s后打印
sleeping 10s
# 打印sleeping 10s后立即打印
hello world!
tokio::spawn方法的返回值是一个handle,如果对这个handle执行.await方法,会阻塞当前调用这个spawn方法的线程,只有在这个handle执行完成后,才会继续执行后面的代码。如下例所示
use tokio;#[tokio::main(flavor = "multi_thread", worker_threads = 2)]
async fn main() {let _ = tokio::task::spawn(async {std::thread::sleep(std::time::Duration::from_secs(10));println!("sleeping 10s");}).await;let _ = tokio::task::spawn(async {std::thread::sleep(std::time::Duration::from_secs(1));println!("second spawn!");}).await;println!("hello world!");
}
在该例子中,因为在主线程中,使用tokio::task::spawn创建了第一个handle_1,并调用该handle_1的.await方法,此时主线程阻塞在这里,等待handle_1执行完毕,即sleep 10s后打印sleep 10s。随后使用tokio::task::spawn创建第二个handle_2,并调用该handle_2的.await方法,此时主线程阻塞在这里,等待handle_2执行完毕,即sleep 1s后打印second spawn!。最后执行主线程中的hello world打印。
上例的输出为:
# 等10s后打印
sleeping 10s
# 打印sleeping 10s完成后等1s后打印
second spawn!
# 打印second spawn完成后立即打印
hello world!
如果我们工作线程有两个,但是我们有3个异步操作会发生什么呢?见下例:
use tokio;#[tokio::main(flavor = "multi_thread", worker_threads = 2)]
async fn main() {let handle_1 = tokio::task::spawn(async {std::thread::sleep(std::time::Duration::from_secs(10));println!("sleeping 10s");});let handle_2 = tokio::task::spawn(async {std::thread::sleep(std::time::Duration::from_secs(1));println!("second spawn!");});let handle_3 = tokio::task::spawn(async {std::thread::sleep(std::time::Duration::from_secs(5));println!("third spawn!");});tokio::join!(handle_1, handle_2, handle_3);println!("hello world!");
}
我们有3个异步spawn,分别睡眠10s、1s、5s。tokio::join按照写代码的顺序,先join的10s的handle_1,然后1s的handle_2,然后5s的handle_3。由于只有两个工作线程,所以10s的handle_1和1s的handle_2可以同时在两个工作线程中执行,但是handle_3不行,因为已经没有多余的工作线程可供handle_3去运行了,所以handle_3只能先挂起,并不执行。过了1s后,当handle_2的工作线程把handle_2执行完,此时这个工作线程空闲出来就可以去执行handle_3了,所以在handle_2的second spawn
打印完成以后,handle_3开始执行,所以最终的输出如下:
# 等1s后打印
second spawn!
# 在second spawn打印后,等5s打印
third spawn!
# 在third spawn打印后,等4s打印
sleeping 10s
# 在sleeping 10s打印后,立即打印hello world
hello world!
tokio::task::spawn_blocking()方法解析
我们知道,当我们定义tokio的时候,可以定义工作线程的数量
#[tokio::main(flavor = "multi_thread", worker_threads = 2)]
但是我们可以看到,在上述的例子中,如果工作线程被阻塞了,即使这个工作线程啥都不做,他也会阻塞在那里,这样CPU就开始摸鱼了,但是我们是社会主义,怎么能摸鱼呢,所以我们不能让CPU有能摸鱼的机会,那么这些阻塞的工作应该怎么办呢?这里tokio给出了一个spawn_blocking的方法。
spawn_blocking方法中的内容,不会在工作线程中运行,而是创建了一个单独的线程用来执行写在spawn_blocking方法中的内容,这样即使是写在spawn_blocking方法中的内容是阻塞的工作,也仅仅是阻塞了新创建出来的这个线程,不会导致用来进行异步操作的工作线程阻塞,这样工作线程可以正常调度其他的各种tokio::spawn而不至于陷在那里。
请看下例:
use tokio;#[tokio::main(flavor = "multi_thread", worker_threads = 2)]
async fn main() {let handle_1 = tokio::task::spawn(async {std::thread::sleep(std::time::Duration::from_secs(10));println!("sleeping 10s");});let handle_2 = tokio::task::spawn(async {std::thread::sleep(std::time::Duration::from_secs(1));println!("second spawn!");});let handle_3 = tokio::task::spawn_blocking(async {std::thread::sleep(std::time::Duration::from_secs(5));println!("third spawn!");});tokio::join!(handle_1, handle_2, handle_3);println!("hello world!");
}
相关文章:
rust语言tokio库spawn, blocking_spawn等的使用
目录 tokio的spawn以及spawn_blocking的使用tokio::task::spawn方法解析tokio::task::spawn_blocking()方法解析 时间会遗忘一切 最后更新时间2024.04.29 tokio版本: tokio的spawn以及spawn_blocking的使用 tokio::task::spawn方法解析 tokio的实现原理以及源码…...

Day_1
1. 环境搭建 技术选型 后端项目结构 sky-take-out maven父工程,统一管理依赖版本,聚合其他子模块 sky-common 子模块,存放公共类,例如:工具类、常量类、异常类等 sky-pojo 子模块,存放实体类、VO、DTO…...

2024.05.06作业
自由发挥应用场景,实现登录界面。 要求:尽量每行代码都有注释。 #include "yuanshen.h"yuanshen::yuanshen(QWidget *parent): QWidget(parent) {//窗口相关设置this->resize(1600,910);this->setFixedSize(1600,910);//窗口标题this-…...

什么是抖音橱窗?它和抖音小店有什么区别?普通人更适合做哪个?
大家好,我是电商糖果 相信有很多想在抖音卖货的朋友,都会搞不清抖音橱窗是什么? 甚至会把它和抖音小店当成一个项目,也不知道哪个更适合自己。 自己越了解发现越迷糊,有的说不需要直播,粉丝,…...

spring高级篇(九)
boot的执行流程分为构造SpringApplication对象、调用run方法两部分 1、Spring Boot 执行流程-构造 通常我们会在SpringBoot的主启动类中写以下的代码: 参数一是当前类的字节码,参数二是main的args参数。 public class StartApplication {public static…...
用wordpress建跨境电商独立站的5大优势
免费和开源 WordPress是一个免费的开源内容管理系统,用户可以自由下载、安装和使用,无需支付版权费用或订阅费用。开源特性也意味着用户可以根据自己的需求修改和定制代码,或者使用其他开发者提供的插件和主题来扩展和美化网站。 易用和灵活…...

Windows中安装的PostgreSQL 数据库如何重启
1. 使用Windows服务管理器 打开“运行”对话框(按WinR键)。输入services.msc并按回车,这将打开服务列表。在服务列表中找到PostgreSQL服务。它通常命名为“PostgreSQL”后面跟着版本号和实例名称,例如“PostgreSQL 13 - mydb”。…...

Remix框架实现 SSR
SSR SSR是一种网页渲染方式,它与传统的客户端渲染(CSR)相对,在日常的项目中我们更多是使用 CSR 的方式进行前端分离开发,渲染会在浏览器端进行。然而在SSR中,当用户请求一个网页时,服务器将生成…...
如何快速开发项目,提高开发效率
文章目录 一、问题描述二、问题解决1.需求分析2.架构设计3.技术选型4.正式开发 一、问题描述 有很多小伙伴在开发一个项目的时候,总是需要很长时间,效率很低,其实本质是没有掌握开发项目的关键和技巧 我下面列举一些问题,不知道…...

面试笔记——多线程使用场景
线程池使用场景(CountDownLatch, Future) CountDownLatch CountDownLatch(闭锁/倒计时锁)用来进行线程同步协作,等待所有线程完成倒计时(一个或者多个线程,等待其他多个线程完成某件…...
02.0 基于Verilog控制LED灯每秒钟闪烁一次
本段代码是为Verilog初学者提供的一个名为led_blink简单实例Verilog模块,其功能是控制6个LED灯同步闪烁,每秒钟闪烁一次。 本例代码用于理解时序逻辑的概念,理解多个always模块完全并行执行的概念,讲授时可以与C语言的执行过程进行…...
C语言创建文件夹和多级目录
C调用系统命令创建多级目录 #include <stdio.h> #include <stdlib.h>int main() {const char *path "a/b/c";// 创建目录命令的字符串char mkdir_command[100];sprintf(mkdir_command, "mkdir %s", path);// 调用系统命令system(mkdir_comma…...

2024.5.6
#include "mywidget.h"MyWidget::MyWidget(QWidget *parent): QWidget(parent) {//设置窗口大小this->resize(1000,740);//设置窗口图标this->setWindowIcon(QIcon("C:\\Users\\Administrator\\Desktop\\pictrue\\Plants.png"));//设置窗口标题this-…...
mybatis配置获取自增主键
mybatis配置获取自增主键 【/n】 01 使用场景 当需要刚刚插入数据库的数据对应的新增主键时,通过配置xml文件,使数据库返回新增主键id,并把主键id与类参数对应 02 涉及配置 注解TableId(type IdType.AUTO):在类主键id通过配…...

完整、免费的把pdf转word文档
在线工具网 https://www.orcc.online 支持pdf转word,免费、完整、快捷 登录网站 https://orcc.online 选择需要转换的pdf文件: 等待转换完成 点击蓝色文件即可下载 无限制,完整转换。...
使用 Lua 协程模拟 Golang 的 go defer 编程模式
封装 go 函数 在 使用 Lua 协程处理异步回调函数 中已经介绍 这里简要列下: 封装 go 函数---go 函数创建并启动一个协程 ---param _co_task function 函数原型 fun(_co:thread) function go(_co_task)local co coroutine.create(_co_task) -- 创建的协程是暂停的…...
网络通信协议,UDP和TCP,初步了解
UDP(User Datagram Protocol)和TCP(Transmission Control Protocol)是两种常见的网络通信协议,用于在计算机网络中进行数据传输。 1. TCP:Transmission Control Protocol(传输控制协议…...

Golang | Leetcode Golang题解之第61题旋转链表
题目: 题解: func rotateRight(head *ListNode, k int) *ListNode {if k 0 || head nil || head.Next nil {return head}n : 1iter : headfor iter.Next ! nil {iter iter.Nextn}add : n - k%nif add n {return head}iter.Next headfor add > …...

美业SaaS系统多门店收银系统源码-【分润常见问题】讲解(一)
美业管理系统源码 博弈美业SaaS系统 连锁多门店美业收银系统源码 多门店管理 / 会员管理 / 预约管理 / 排班管理 / 商品管理 / 促销活动 PC管理后台、手机APP、iPad APP、微信小程序 ▶ 分润常见问题: 1、分润金额基数 所有分润计算的基数均为平台订单中各个商…...

Chatbot 在教育中的应用
Chatbot 在教育中的应用 基本信息 这篇博客主要介绍几篇Chatbot在教育领域中应用的文章,根据文章的侧重点不同,分为介绍教育理论,与介绍系统设计两类。从问题定义、技术方法、教育学理论、实验设计、结论证据几个方面概括各篇文章。 博…...

idea大量爆红问题解决
问题描述 在学习和工作中,idea是程序员不可缺少的一个工具,但是突然在有些时候就会出现大量爆红的问题,发现无法跳转,无论是关机重启或者是替换root都无法解决 就是如上所展示的问题,但是程序依然可以启动。 问题解决…...

日语AI面试高效通关秘籍:专业解读与青柚面试智能助攻
在如今就业市场竞争日益激烈的背景下,越来越多的求职者将目光投向了日本及中日双语岗位。但是,一场日语面试往往让许多人感到步履维艰。你是否也曾因为面试官抛出的“刁钻问题”而心生畏惧?面对生疏的日语交流环境,即便提前恶补了…...
java_网络服务相关_gateway_nacos_feign区别联系
1. spring-cloud-starter-gateway 作用:作为微服务架构的网关,统一入口,处理所有外部请求。 核心能力: 路由转发(基于路径、服务名等)过滤器(鉴权、限流、日志、Header 处理)支持负…...
golang循环变量捕获问题
在 Go 语言中,当在循环中启动协程(goroutine)时,如果在协程闭包中直接引用循环变量,可能会遇到一个常见的陷阱 - 循环变量捕获问题。让我详细解释一下: 问题背景 看这个代码片段: fo…...
深入浅出:JavaScript 中的 `window.crypto.getRandomValues()` 方法
深入浅出:JavaScript 中的 window.crypto.getRandomValues() 方法 在现代 Web 开发中,随机数的生成看似简单,却隐藏着许多玄机。无论是生成密码、加密密钥,还是创建安全令牌,随机数的质量直接关系到系统的安全性。Jav…...

如何将联系人从 iPhone 转移到 Android
从 iPhone 换到 Android 手机时,你可能需要保留重要的数据,例如通讯录。好在,将通讯录从 iPhone 转移到 Android 手机非常简单,你可以从本文中学习 6 种可靠的方法,确保随时保持连接,不错过任何信息。 第 1…...

Mac软件卸载指南,简单易懂!
刚和Adobe分手,它却总在Library里给你写"回忆录"?卸载的Final Cut Pro像电子幽灵般阴魂不散?总是会有残留文件,别慌!这份Mac软件卸载指南,将用最硬核的方式教你"数字分手术"࿰…...
C++.OpenGL (10/64)基础光照(Basic Lighting)
基础光照(Basic Lighting) 冯氏光照模型(Phong Lighting Model) #mermaid-svg-GLdskXwWINxNGHso {font-family:"trebuchet ms",verdana,arial,sans-serif;font-size:16px;fill:#333;}#mermaid-svg-GLdskXwWINxNGHso .error-icon{fill:#552222;}#mermaid-svg-GLd…...
CRMEB 框架中 PHP 上传扩展开发:涵盖本地上传及阿里云 OSS、腾讯云 COS、七牛云
目前已有本地上传、阿里云OSS上传、腾讯云COS上传、七牛云上传扩展 扩展入口文件 文件目录 crmeb\services\upload\Upload.php namespace crmeb\services\upload;use crmeb\basic\BaseManager; use think\facade\Config;/*** Class Upload* package crmeb\services\upload* …...
JS设计模式(4):观察者模式
JS设计模式(4):观察者模式 一、引入 在开发中,我们经常会遇到这样的场景:一个对象的状态变化需要自动通知其他对象,比如: 电商平台中,商品库存变化时需要通知所有订阅该商品的用户;新闻网站中࿰…...