C++23中if consteval / if not consteval (P1938R3) 详解
文章目录
- 引言
- 基本概念
- `consteval` 回顾
- `if consteval` 和 `if not consteval` 的定义
- 语法规则
- 设计目的
- 解决现有问题
- 增强代码的可读性和可维护性
- 提高代码的性能和安全性
- 使用示例
- 简单示例
- 复杂示例
- 与其他特性的对比
- 与 `if constexpr` 的对比
- 与 `std::is_constant_evaluated()` 的对比
- 实际应用场景
- 编译时反射
- 数学计算
- 模板元编程
- 类型安全操作
- 总结
引言
在C++的发展历程中,编译时计算一直是一个重要的特性,它可以提高程序的性能和安全性。C++20引入了 consteval
和 std::is_constant_evaluated()
等特性,前者用于声明必须在编译期间完成调用的立即函数,后者用于检查当前是否处于常量求值上下文。而在C++23中,进一步引入了 if consteval
和 if not consteval
语法,使得在代码中区分编译时和运行时行为变得更加方便和灵活。本文将详细介绍 if consteval
和 if not consteval
的相关内容,包括语法规则、设计目的、实际应用场景等。
基本概念
consteval
回顾
在介绍 if consteval
和 if not consteval
之前,我们先回顾一下 consteval
的基本概念。consteval
用于声明立即函数,这类函数的调用必须在编译期间完成。如果尝试在运行时调用 consteval
函数,会导致编译错误。它的目标是确保某些逻辑完全在编译时执行,适用于需要强制编译期计算的场景。例如:
consteval int square(int x) {return x * x;
}constexpr int a = square(5); // 编译时调用
// int b = square(rand()); // 错误:运行时调用
if consteval
和 if not consteval
的定义
if consteval
是C++23引入的编译时条件构造,它允许开发者编写仅在常量求值上下文中执行的代码。而 if not consteval
则是其否定形式,用于编写在非常量求值上下文中执行的代码。
语法规则
if consteval
和 if not consteval
的语法如下:
语法形式 | 描述 |
---|---|
if consteval compound-statement | 如果当前处于常量求值上下文,则执行 compound-statement |
if consteval compound-statement else statement | 如果当前处于常量求值上下文,则执行 compound-statement ;否则执行 statement |
if not consteval compound-statement | 如果当前处于非常量求值上下文,则执行 compound-statement |
if not consteval compound-statement else statement | 如果当前处于非常量求值上下文,则执行 compound-statement ;否则执行 statement |
其中,compound-statement
是复合语句,通常用花括号 {}
括起来;statement
可以是单个语句或复合语句。
设计目的
解决现有问题
在C++20中,虽然有 std::is_constant_evaluated()
可以用于检查当前是否处于常量求值上下文,但使用起来存在一些问题。例如,开发者可能会在运行时进行判断,导致出现“对即时函数的调用不是常量表达式”的错误。而 if consteval
和 if not consteval
则提供了一种更直接、更安全的方式来区分编译时和运行时行为。
增强代码的可读性和可维护性
使用 if consteval
和 if not consteval
可以使代码更加清晰地表达哪些代码是在编译时执行,哪些是在运行时执行,从而提高代码的可读性和可维护性。
提高代码的性能和安全性
通过在编译时执行一些逻辑,可以避免运行时的开销,提高程序的性能。同时,强制某些操作在编译时完成,也可以增强代码的类型安全性。
使用示例
简单示例
#include <iostream>consteval int compile_time_abs(int x) {return x < 0 ? -x : x;
}constexpr int abs(int x) {if consteval { // 仅在编译时求值时执行此分支return compile_time_abs(x);} else { // 运行时调用return x < 0 ? -x : x;}
}int main() {constexpr int a = abs(-5); // 编译时调用,使用 if consteval 分支int b = abs(-10); // 运行时调用,使用 else 分支std::cout << "a: " << a << ", b: " << b << std::endl;return 0;
}
在这个示例中,abs
函数根据当前是否处于常量求值上下文,选择不同的执行路径。如果是编译时调用,则使用 compile_time_abs
函数;如果是运行时调用,则直接计算绝对值。
复杂示例
#include <cmath>
#include <cstdint>
#include <iostream>constexpr bool is_constant_evaluated() noexcept {if consteval { return true; } else { return false; }
}constexpr bool is_runtime_evaluated() noexcept {if not consteval { return true; } else { return false; }
}consteval std::uint64_t ipow_ct(std::uint64_t base, std::uint8_t exp) {if (!base) return base;std::uint64_t res{1};while (exp) {if (exp & 1) res *= base;exp /= 2;base *= base;}return res;
}constexpr std::uint64_t ipow(std::uint64_t base, std::uint8_t exp) {if consteval { // 使用编译时友好的算法return ipow_ct(base, exp);} else {return std::pow(base, exp);}
}int main() {constexpr std::uint64_t c = ipow(2, 10); // 编译时调用std::uint64_t r = ipow(3, 5); // 运行时调用std::cout << "c: " << c << ", r: " << r << std::endl;return 0;
}
在这个示例中,ipow
函数根据当前是否处于常量求值上下文,选择不同的幂运算算法。如果是编译时调用,则使用 ipow_ct
函数;如果是运行时调用,则使用 std::pow
函数。
与其他特性的对比
与 if constexpr
的对比
if constexpr
是C++17引入的特性,用于在编译时进行条件判断。它的条件表达式必须是一个常量表达式,并且会在编译时根据条件的值选择执行哪个分支,未被选择的分支会被丢弃。而 if consteval
则是根据当前是否处于常量求值上下文来选择执行路径,它可以在运行时进行判断。例如:
template <typename T>
constexpr auto get_value() {if constexpr (std::is_integral_v<T>) {return 42;} else {return 3.14;}
}constexpr int a = get_value<int>(); // 编译时选择分支
与 std::is_constant_evaluated()
的对比
std::is_constant_evaluated()
是C++20引入的库函数,用于检查当前是否处于常量求值上下文。但使用它时需要注意,可能会在运行时进行判断,导致一些意外的错误。而 if consteval
和 if not consteval
则提供了更直接、更安全的方式来区分编译时和运行时行为。例如:
constexpr int Add(std::span<const int> sp) {if (std::is_constant_evaluated()) {// 可能在运行时判断}if consteval {// 仅在编译时判断}
}
实际应用场景
编译时反射
在编译时反射中,需要生成类型相关的元信息。使用 if consteval
可以确保这些操作在编译时完成,提高程序的性能。例如:
// 伪代码示例
template <typename T>
consteval auto get_type_info() {// 编译时生成类型信息return ...;
}template <typename T>
constexpr auto print_type_info() {if consteval {auto info = get_type_info<T>();// 处理类型信息}
}
数学计算
对于需要确保编译时优化的常量计算,使用 if consteval
可以避免运行时的开销。例如:
consteval int factorial(int n) {return n <= 1 ? 1 : n * factorial(n - 1);
}constexpr int a = factorial(5); // 编译时计算
模板元编程
if consteval
可以替代部分模板元编程逻辑,简化代码。例如:
template <typename T>
constexpr auto process(T value) {if consteval {// 编译时处理逻辑} else {// 运行时处理逻辑}
}
类型安全操作
通过强制某些类型转换在编译时完成,可以增强代码的类型安全性。例如:
template <typename T>
constexpr auto convert(T value) {if consteval {// 编译时类型转换return static_cast<...>(value);} else {// 运行时类型转换return ...;}
}
总结
C++23中引入的 if consteval
和 if not consteval
语法,为开发者提供了一种更直接、更安全的方式来区分编译时和运行时行为。它可以提高代码的可读性、可维护性、性能和安全性,适用于编译时反射、数学计算、模板元编程、类型安全操作等多种场景。在实际开发中,建议结合最新编译器(如GCC 13+、Clang 16+或MSVC 2022)体验这些特性。
相关文章:
C++23中if consteval / if not consteval (P1938R3) 详解
文章目录 引言基本概念consteval 回顾if consteval 和 if not consteval 的定义语法规则 设计目的解决现有问题增强代码的可读性和可维护性提高代码的性能和安全性 使用示例简单示例复杂示例 与其他特性的对比与 if constexpr 的对比与 std::is_constant_evaluated() 的对比 实…...
IPOF(Input-Process-Output-Feedback)方法学简介
一、背景知识:IPO与IPOF的基本概念 1. IPO(Input-Process-Output,输入-处理-输出) IPO是计算机科学、系统工程和设计领域中最基础的模型之一,用于描述系统如何将输入通过处理逻辑转换为输出。其核心是线性的单向流程…...
第十二天 使用Unity Test Framework进行自动化测试 性能优化:Profiler分析、内存管理
前言 在完成游戏核心功能开发后,如何确保项目质量并成功发布到各大平台?本文将从自动化测试到商店上架,手把手教你构建完整的游戏开发闭环。使用Unity 2022 LTS版本进行演示,所有代码均经过实际项目验证。 一、自动化测试实战&am…...
图解YOLO(You Only Look Once)目标检测(v1-v5)
1. YOLO系列整体介绍 YOLO属于深度学习经典检测方法中的单阶段(one - stage)类型,与两阶段(two - stage,如Faster - rcnn、Mask - Rcnn系列)方法相对。 不同模型性能 单阶段方法的最核心优势是速度非常快…...
U8G2在PC端模拟(C语言版本)
前提: 电脑已经准备好mingw编译器环境,已经加入环境变量. 测试方法: window下打开cmd,输入gcc -v 会有信息打印. u8g2 u8g2官方支持sdl2接口,已经做好了适配. 所以只需要在使用的开发环境配置好SDL2路径即可. sdl2和u8g2的适配…...
mac brew 无法找到php7.2 如何安装php7.2
mac brew 无法找到php7.2 如何安装php7.2 原因是升级过高版本的brew后已经不支持7.2了,但可以通过第三方工具来安装 brew tap shivammathur/php brew install shivammathur/php/php7.2标题安装完成后会提示以下信息: The php.ini and php-fpm.ini fil…...
苹果新规生效:即日起不再接受iOS 17 SDK编译的应用提交
今天(2025年4月25日),苹果公司正式实施了一项重要的开发者政策变更:所有提交至App Store Connect的应用必须使用iOS 18 SDK或更高版本编译,这意味着基于iOS 17.2及更早版本SDK构建的应用将无法再被提交审核。 这一错误…...
OpenCv高阶(九)——背景建模
目录 一、背景建模的核心目标与核心挑战 1. 核心目标 2. 核心挑战 二、背景建模模型 1、帧差法原理 2. 概率模型(Parametric Models) (1)高斯混合模型(Gaussian Mixture Model, GMM) (…...
Docker Macvlan网络配置实战:解决“network already exists“错误
一、Macvlan网络的应用场景 Macvlan是Docker支持的多种网络驱动之一,它允许容器直接绑定到宿主机的物理接口,为每个容器分配独立的MAC地址。这种网络模式特别适合需要容器直接暴露在物理网络中的场景,例如: • IoT设备直接通信 …...
Android——Activity与Fragment通信
Activity向Fragment传递数据 Activity.java Bundle bundle new Bundle();bundle.putString("message", "wonima");BlankFragment bf new BlankFragment();bf.setArguments(bundle);Fragment.java Overridepublic View onCreateView(LayoutInflater infl…...
38 python random
在实际中,我们常常会用到随机的概念,比如 模拟抽奖活动(如:月度优秀员工抽奖)生成测试数据(如:随机考勤时间、随机销售额)打乱数据顺序(如:随机分配任务到人)Python 的random模块就像你的 "随机事件生成器",帮你轻松创建各种随机数据 一、基础操作:从随…...
二叉树的遍历(广度优先搜索)
二叉树的第二种遍历方式,层序遍历,本质是运用队列对二叉树进行搜索。 层序遍历是指将二叉树的每一层按顺序遍历,通过队列实现就是先将根节点push入队,统计此时的队列中的元素数量size,将size元素全部pop出去࿰…...
设备接入与APP(应用程序)接入华为云iotDA平台的详细操作步骤及获取方式
壹、设备登录 IoTDA 的通关清单详细操作路径及获取方式 下面将之前整理的“设备登录 IoTDA 的通关清单”补充为带有详细操作路径的实用指南,逐项说明这些信息在哪里查、怎么查、怎么用,可以一步步操作落地,无需额外查文档。 ✅【完整版】设备…...
100个用户的聊天系统:轮询 vs WebSocket 综合对比
📊 对比表 对比维度普通轮询(Polling)WebSocket实时性⏳ 一般(延迟轮询间隔)例如 5 秒轮询,平均延迟 2.5 秒⚡️ 高(消息可毫秒级送达)数据库压力🚨 高(每次…...
第四章第四节 Spark-Streaming核心编程(三)
打开kafka集群 Spark Streaming支持多种数据源接入方式,有两种典型场景 基于Filebeat的目录监控:通过Filebeat实时监控指定目录(如日志目录),将新增文件内容采集并推送至Kafka消息队列。这种方式适用于需要持久化数据的…...
IDEA编写flinkSQL(快速体验版本,--无需配置环境)
相关资料 文档内容链接地址datagen生成器https://nightlies.apache.org/flink/flink-docs-release-1.16/docs/connectors/table/datagen/print 生成器https://nightlies.apache.org/flink/flink-docs-release-1.16/docs/connectors/table/print/ 准备工作 优点就是下载个ide…...
树莓派4B+Ubuntu24.04 电应普超声波传感器串口输出 保姆级教程
1. 背景 我的机器人上要用到超声波测距,因此就选了电应普的1拖4的超声波测距模块。用的是UART受控输出。这个模块的使用很简单,这里做一个笔记,也是对大家的一个参考。 2. 保姆级教程 刚刚说了,这东西很简单,就是一…...
Aloudata Agent :基于 NoETL 明细语义层的分析决策智能体
今天的分享包含四方面内容。首先,谈谈企业在数据分析智能体(Agent)探索时所面临的挑战;其次,介绍 Aloudata Agent 的关键技术创新点;第三,作为首次亮相,我们将通过产品演示揭晓产品 …...
C++学习-入门到精通-【1】C++编程入门,输入/输出和运算符
C学习-入门到精通-【1】C编程入门,输入/输出和运算符 C编程入门,输入/输出和运算符 C学习-入门到精通-【1】C编程入门,输入/输出和运算符第一个C程序:输出一行文本算术运算 第一个C程序:输出一行文本 // 文本打印程序…...
力扣-234.回文链表
题目描述 给你一个单链表的头节点 head ,请你判断该链表是否为回文链表。如果是,返回 true ;否则,返回 false 。 class Solution { public:bool isPalindrome(ListNode* head) {//快慢指针找到中间结点p1(偶数个结点…...
Electron从入门到入门
项目说明 项目地址 项目地址:https://gitee.com/ruirui-study/electron-demo 本项目为示例项目,代码注释非常清晰,给大家当做入门项目吧。 其实很多东西都可以在我这基础上添加或修改、市面上有些已开源的项目,但是太臃肿了&am…...
【华为HCIP | 华为数通工程师】821—多选解析—第十七页
多选835、IS-IS协议所使用的NSAP地址主要由哪几个部分构成? A、AREA ID B、SEL C、DSCp D、SYSTEM ID 解析:NSAP地址:网络服务访问点(Network Service Access Point)是 OSI 协议中用于定位资源的地址。NSAP 的地址结构如图所示,它由 IDP(Initial Domain …...
electron-builder 打包安装与启动手动安装,最终解决方案,之前的文章与其他的人都不用看了。
介绍 最近官网出了个electron-forge,但打包之后的软件包有问题会有个多余的遮罩物挡在软件的最前面,太恶心了。研究了很久硬是没法去掉。没办法只能重温electron-builder工具来进行打包。之前用过这个安装过一次,也是耗费了好几天。而且也做好了笔记,有资料的情况下,我万万…...
conda安装cuda+cudnn+pytorch【一条龙服务,以cuda12.1版本为例】
安装步骤在Windows和ubuntu系统中应该是一致的,建议浏览一遍文章再尝试配置环境 一、创建新的虚拟环境 这里以Anaconda为例,打开终端 创建新的虚拟环境,test_torch可替换为别的名字(自定义),3.10也可以改…...
ElementUi的tabs样式太难修改,自定义tabs标签页
ElementUi的Tabs组件在某些情况下难以是自己想要的样式,这时候自定义 Tabs 会是一个更好的选择,可以根据自己想要而设置样式,如图: 一、ElementUi的Tabs样式 链接:Tabs 标签页 | Element Plus 基础: 选…...
制作一款打飞机游戏23:编辑器ui
今天,我想重点介绍编辑器的实际用户界面(UI)。 关于Excel的讨论 很多人使用Excel来编辑他们程序的数据,因为大多数时候它都能很好地完成工作。Excel就像是一把瑞士军刀,可以修改各种数据。但是,在某些情况…...
课程9. 机器翻译,Seq2Seq与Attention
课程9. 机器翻译,Seq2Seq与Attention 机器翻译的任务. Seq2Seq 架构通过实战理解加载和预处理数据构建 Seq2Seq 模型编码器解码器Seq2Seq网络训练 Seq2Seq 架构问题注意力机制(Attention)注意选项Transformer 架构介绍——BERT 课程计划 机器…...
ASP.NET MVC 入门指南二
9. 表单处理与提交 9.1 创建表单视图 在视图文件夹下创建一个用于创建产品的视图,如 Create.cshtml: html model YourNamespace.Product{ViewBag.Title "创建产品"; }<h2>创建产品</h2>using (Html.BeginForm()) {Html.Anti…...
JavaWeb学习打卡-Day3-MyBatis相关
MyBatis 什么是MyBatis? MyBatis是一款优秀的持久层框架,用于简化JDBC的开发。MyBatis免除了几乎所有的 JDBC 代码以及设置参数和获取结果集的工作。 JDBC JDBC(Java DataBase Connectivity):使用Java语言操作关系型数据库的…...
浅谈AI Agent 演进之路
1、了解下 AI Agent 的定义 AI Agent(人工智能代理)简单来说是一种能够感知环境、进行决策和执行动作的智能实体。与传统的人工智能相比,AI Agent 具备独立思考和调用工具逐步完成目标的能力。 例如:当要求 AI Agent 帮助下单外…...
佳博票据和标签打印:Web网页端与打印机通信 | iOS
文章目录 引言I Web网页端与打印机通信webSDK(包含示例页)打印测试II iOS与佳博打印机通信引言 佳博工具下载ESC是票据打印指令,TSC是标签打印指令 工业打印机:佳博GP-H430F工业机标签条码打印机物流快递电子面单条码机碳带机 应用场景:打印商品价格标签、打印交易小票 I…...
视频噪点多,如何去除画面噪点?
你是否遇到过这样的困扰?辛辛苦苦拍摄的视频,导出后却满屏 “雪花”,夜景变 “噪点盛宴”,低光环境秒变 “马赛克现场”? 无论是日常拍摄的vlog、珍贵的家庭录像,还是专业制作的影视作品,噪点问…...
微信小程序直传阿里云 OSS 实践指南(V4 签名 · 秒传支持 · 高性能封装)
文章目录 前言一、为什么要使用直传 OSS?二、整体架构与实现思路三、阿里云 OSS 配置(V4 签名)1. 权限设置2. 后端生成签名参数(返回给小程序) 四、微信小程序端上传流程(功能模块拆解与封装)第…...
云原生--核心组件-容器篇-1-Docker和云原生关系(Docker是云原生的基石)
1、基本概念 (1)、云原生(Cloud Native) 是一种构建和运行应用程序的方法论,旨在充分利用云计算环境(公有云、私有云、混合云)的特性,通过容器化、微服务、服务网格、声明式API等技…...
GAEA情感坐标背后的技术原理
基于GAEA的去中心化物理基础设施网络(DePIN),用户有机会在GAEA平台上获得宝贵的数据共享积分。为了提升这些洞察的丰富性,用户必须花费一定数量的积分,将过去的网络数据与当前的情感数据绑定,从而产生一种新…...
day01_编程语言介绍丶Java语言概述丶开发环境搭建丶常用DOS命令
编程语言介绍 编程语言是一种用于人与计算机之间通信的语言,允许程序员编写代码,这些代码告诉计算机要执行哪些操作。编程语言可以被视为计算机可以理解并执行的指令集合,它是一种标准化的交流技巧,用于向计算机发出指令。…...
STM32系列官方标准固件库的完整下载流程
一、官网导航与版本确认 访问ST官网 打开浏览器进入 ST官网,点击左侧 “工具与软件” 标签,展开后选择 “嵌入式软件” 。若页面未直接显示,可在搜索框输入“STM32 Standard Peripheral Libraries”查找。 → “STM32标准外设库”࿰…...
Android 14 系统统一修改app启动时图标大小和圆角
Android 14 统一修改app启动时图标大小和圆角 修改如下: 目录:frameworks/base/core/java/android/window/SplashScreenView.java frameworks/base/libs/WindowManager/Shell/src/com/android/wm/shell/startingsurface/SplashscreenContentDrawer.ja…...
MySQL 详解之函数:数据处理与计算的利器
在 MySQL 中,函数可以接受零个或多个输入参数,并返回一个值。这些函数可以在 SELECT 语句的字段列表、WHERE 子句、HAVING 子句、ORDER BY 子句以及 UPDATE 和 INSERT 语句中使用。合理利用函数,可以简化 SQL 语句,提高开发效率。 MySQL 提供了大量的内置函数 (Built-in F…...
Tailwind CSS 实战:基于 Kooboo 构建企业官网页面(一)
目录 一、技术选型:为什么选择Tailwind Kooboo? 二、CDN方案 vs 传统安装 三、CDN方式实战步骤 一、技术选型:为什么选择Tailwind Kooboo? 1.1 黄金组合优势 Tailwind CSS:原子化CSS框架,提供&#x…...
Java基础 — 条件结构与随机数
介绍 Java条件结构与随机数是程序逻辑控制的重要工具。条件结构通过if-else和switch实现分支判断:if(条件){代码}用于单分支,else if添加多条件判断,switch则基于固定值匹配不同case。随机数生成常用两种方式:Math.random()方法返…...
AI网络渗透kali应用(gptshell)
kali安装gptshell 一、shellGPT 工具介绍 ShellGPT是一款由AI大型语言模型(LLM)驱动的终端命令行工具。它能帮助用户直接在终端与AI交互,自动生成、解释、执行各类 Linux 命令,大大提升了运维和开发效率。ShellGPT 支持接入 O…...
如何实现Android屏幕和音频采集并启动RTSP服务?
技术背景 在移动直播和视频监控领域,实现高效的屏幕和音频采集并提供流媒体服务是关键技术之一。本文将详细介绍如何基于大牛直播SDK实现Android屏幕和麦克风/扬声器采集,并启动轻量级RTSP服务以对外提供拉流的RTSP URL。在Android平台上,轻…...
大模型提示词如何编写
一、提示词的核心三要素 明确目标(What) 告诉 AI「你要它做什么」,越具体越好。 ❌ 模糊:写一篇文章 ✅ 清晰:写一篇 800 字的高考作文,主题 “坚持与创新”,结构分引言、三个论点(…...
Serverless 在云原生后端的实践与演化:从函数到平台的革新
📝个人主页🌹:慌ZHANG-CSDN博客 🌹🌹期待您的关注 🌹🌹 一、引言:从服务器到“无服务器”的后端演变 在传统后端开发中,我们需要为服务配置并维护服务器资源,无论是物理机、虚拟机还是容器化服务,都需要: 管理系统运行环境 监控负载与扩缩容 保证高可用与安…...
反爬虫机制中的验证码识别:类型、技术难点与应对策略
在互联网数据抓取领域,验证码识别是爬虫过程中的关键环节之一。下面对常见验证码类型、技术难点及应对策略进行详细解析,并提供多种场景下的代码实现示例。 一、验证码类型与技术难点 (一)图形验证码 1. 字符验证码 特征&#…...
.NET 10 中的新增功能
.NET 运行时 .NET 10 运行时引入了新功能和性能改进。 关键更新包括: 数组接口方法反虚拟化:JIT 现在可以取消虚拟化和内联数组接口方法,从而提高数组枚举的性能。数组枚举去抽象化:改进功能以通过枚举器减少数组迭代的抽象开销…...
通过音频的pcm数据格式利用canvas绘制音频波形图
上面是一个完整的音频的波形图,可以大概知道音频整个的简略信息 数据准备:需要有这个音频的pcm数据,也就是时域采样值,每个数字代表某一时刻音频波形的振幅。 <!DOCTYPE html> <html lang"en"> <head&…...
Dubbo负载均衡策略深度解析
互联网大厂Java求职者面试:Dubbo负载均衡策略详解 第一轮提问: 面试官:马架构,您好!请问您了解Dubbo的负载均衡策略吗?Dubbo支持哪些负载均衡策略呢? 马架构:您好!Dub…...
【高频考点精讲】async/await原理剖析:Generator和Promise的完美结合
async/await原理剖析:Generator和Promise的完美结合 今天咱们聊聊async/await,这玩意儿用起来是真香,但你知道它背后是怎么运作的吗?其实它就是Generator和Promise的"爱情结晶"。 1. 先搞懂Generator Generator&…...