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

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 使用场景 当需要刚刚插入数据库的数据对应的新增主键时&#xff0c;通过配置xml文件&#xff0c;使数据库返回新增主键id&#xff0c;并把主键id与类参数对应 02 涉及配置 注解TableId(type IdType.AUTO)&#xff1a;在类主键id通过配…...

完整、免费的把pdf转word文档

在线工具网 https://www.orcc.online 支持pdf转word&#xff0c;免费、完整、快捷 登录网站 https://orcc.online 选择需要转换的pdf文件&#xff1a; 等待转换完成 点击蓝色文件即可下载 无限制&#xff0c;完整转换。...

使用 Lua 协程模拟 Golang 的 go defer 编程模式

封装 go 函数 在 使用 Lua 协程处理异步回调函数 中已经介绍 这里简要列下&#xff1a; 封装 go 函数---go 函数创建并启动一个协程 ---param _co_task function 函数原型 fun(_co:thread) function go(_co_task)local co coroutine.create(_co_task) -- 创建的协程是暂停的…...

网络通信协议,UDP和TCP,初步了解

UDP&#xff08;User Datagram Protocol&#xff09;和TCP&#xff08;Transmission Control Protocol&#xff09;是两种常见的网络通信协议&#xff0c;用于在计算机网络中进行数据传输。 1. TCP&#xff1a;Transmission Control Protocol&#xff08;传输控制协议&#xf…...

Golang | Leetcode Golang题解之第61题旋转链表

题目&#xff1a; 题解&#xff1a; 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、微信小程序 ▶ 分润常见问题&#xff1a; 1、分润金额基数 所有分润计算的基数均为平台订单中各个商…...

Chatbot 在教育中的应用

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

idea大量爆红问题解决

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

日语AI面试高效通关秘籍:专业解读与青柚面试智能助攻

在如今就业市场竞争日益激烈的背景下&#xff0c;越来越多的求职者将目光投向了日本及中日双语岗位。但是&#xff0c;一场日语面试往往让许多人感到步履维艰。你是否也曾因为面试官抛出的“刁钻问题”而心生畏惧&#xff1f;面对生疏的日语交流环境&#xff0c;即便提前恶补了…...

java_网络服务相关_gateway_nacos_feign区别联系

1. spring-cloud-starter-gateway 作用&#xff1a;作为微服务架构的网关&#xff0c;统一入口&#xff0c;处理所有外部请求。 核心能力&#xff1a; 路由转发&#xff08;基于路径、服务名等&#xff09;过滤器&#xff08;鉴权、限流、日志、Header 处理&#xff09;支持负…...

golang循环变量捕获问题​​

在 Go 语言中&#xff0c;当在循环中启动协程&#xff08;goroutine&#xff09;时&#xff0c;如果在协程闭包中直接引用循环变量&#xff0c;可能会遇到一个常见的陷阱 - ​​循环变量捕获问题​​。让我详细解释一下&#xff1a; 问题背景 看这个代码片段&#xff1a; fo…...

深入浅出:JavaScript 中的 `window.crypto.getRandomValues()` 方法

深入浅出&#xff1a;JavaScript 中的 window.crypto.getRandomValues() 方法 在现代 Web 开发中&#xff0c;随机数的生成看似简单&#xff0c;却隐藏着许多玄机。无论是生成密码、加密密钥&#xff0c;还是创建安全令牌&#xff0c;随机数的质量直接关系到系统的安全性。Jav…...

如何将联系人从 iPhone 转移到 Android

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

Mac软件卸载指南,简单易懂!

刚和Adobe分手&#xff0c;它却总在Library里给你写"回忆录"&#xff1f;卸载的Final Cut Pro像电子幽灵般阴魂不散&#xff1f;总是会有残留文件&#xff0c;别慌&#xff01;这份Mac软件卸载指南&#xff0c;将用最硬核的方式教你"数字分手术"&#xff0…...

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):观察者模式 一、引入 在开发中&#xff0c;我们经常会遇到这样的场景&#xff1a;一个对象的状态变化需要自动通知其他对象&#xff0c;比如&#xff1a; 电商平台中&#xff0c;商品库存变化时需要通知所有订阅该商品的用户&#xff1b;新闻网站中&#xff0…...