C++17模板编程与if constexpr深度解析
一、原理深化
1.1 模板编程
1.1.1 编译器如何处理模板(补充)
模板的实例化机制存在两种模式:
- 隐式实例化:编译器在遇到模板具体使用时自动生成代码,可能导致多翻译单元重复实例化,增加编译时间。
- 显式实例化:通过
template class MyTemplate<int>;
指令强制在指定位置生成代码,可优化编译速度并控制符号可见性。
两阶段查找(Two-Phase Lookup):
- 模板定义阶段:检查非依赖名称(不依赖模板参数的符号),立即进行语法检查。
- 模板实例化阶段:检查依赖名称(依赖模板参数的符号),此时才会进行ADL(参数依赖查找)和完整类型检查。
template<typename T>
void func(T x) {non_dependent(); // 阶段1检查,立即报错若未声明dependent(x); // 阶段2检查,实例化时才检查
}
1.1.2 汇编与链接(补充)
- 符号重复问题:C++标准要求链接器合并等价模板实例,但不同编译器实现差异可能导致ODR(单一定义规则)违规。可通过
inline
或显式实例化避免。 - 模板代码膨胀:多次实例化
vector<int>
和vector<double>
会生成独立代码,可通过模板显式特化或类型擦除技术优化体积。
1.2 if constexpr
(补充)
1.2.1 编译时短路与类型系统
if constexpr
的核心优势在于编译时分支消除,使得被丢弃的分支:
- 不参与类型检查
- 不参与函数重载决议
- 不要求语法合法性(只要不依赖模板参数)
示例对比:
template<typename T>
void process() {if constexpr (false) {T::invalid(); // 允许:分支被丢弃}
}template<typename T>
void process_old() {if (false) {T::invalid(); // 编译错误:即使不执行仍需合法}
}
1.2.2 与SFINAE的协同
在C++17之前,需通过enable_if
实现条件编译:
// C++11风格
template<typename T, typename = std::enable_if_t<std::is_integral<T>::value>>
void func(T t) { /*...*/ }
if constexpr
可简化逻辑:
template<typename T>
void func(T t) {if constexpr (std::is_integral<T>::value) {// 仅整数类型逻辑}
}
二、应用场景扩展
2.1 模板元编程进阶
类型分发与编译时计算:
template<size_t N>
struct Factorial {static constexpr size_t value = N * Factorial<N-1>::value;
};
template<>
struct Factorial<0> {static constexpr size_t value = 1;
};// 使用if constexpr替代部分元编程
template<size_t N>
constexpr size_t factorial() {if constexpr (N == 0) return 1;else return N * factorial<N-1>();
}
2.2 if constexpr
在泛型回调中的应用
处理异构类型容器:
template<typename... Ts>
void processVariant(const std::variant<Ts...>& var) {std::visit([](auto&& arg) {using T = std::decay_t<decltype(arg)>;if constexpr (std::is_same_v<T, int>) {std::cout << "Int: " << arg * 2;} else if constexpr (std::is_same_v<T, std::string>) {std::cout << "Str: " << arg.size();}}, var);
}
三、实践优化与陷阱
3.1 性能对比分析
汇编对比实验:
// 普通if语句
template<typename T>
void func(T t) {if (std::is_integral<T>::value) { /* A */ }else { /* B */ }
}// if constexpr
template<typename T>
void func(T t) {if constexpr (std::is_integral<T>::value) { /* A */ }else { /* B */ }
}
- 当实例化为
func<int>
时,普通if会保留B分支的跳转指令,而if constexpr
完全消除B分支代码。
3.2 常见陷阱
- 依赖作用域:
template<typename T>
void func() {if constexpr (condition) {using Type = int;} else {using Type = double; // 错误:两个分支的Type不在同一作用域}Type value; // 需改为外部定义
}
- 非布尔类型转换:
if constexpr (sizeof(T)) { ... } // 错误:需显式转换为bool
if constexpr (!!sizeof(T)) { ... } // 正确
四、总结扩展
模板与if constexpr
的结合标志着C++向编译时计算泛型化的演进。C++20的Concepts进一步简化约束表达:
template<std::integral T> // C++20概念
void func(T t) {if constexpr (std::signed_integral<T>) { ... }
}
开发者应掌握:
- 模板实例化机制对编译性能的影响
if constexpr
与SFINAE的适用场景取舍- 编译时分支的类型系统行为
通过合理组合这些特性,可构建出类型安全、零开销抽象的高性能代码库。
以下为专业扩展内容,建议有余力再来继续阅读
五、编译器处理模板的汇编细节(以GCC 13为例)
1.1 模板函数实例化的汇编表现
C++代码:
// demo_template.cpp
template<typename T>
T add(T a, T b) { return a + b; }int main() {add<int>(1, 2); // 显式实例化add<double>(3.0, 4.0);
}
生成汇编命令:
g++ -S -O0 demo_template.cpp -o demo_template.s
关键汇编输出(x86_64):
; add<int>实例化
_Z3addIiET_S0_S0_:pushq %rbpmovq %rsp, %rbpmovl %edi, -4(%rbp) ; int amovl %esi, -8(%rbp) ; int bmovl -4(%rbp), %edxaddl -8(%rbp), %edx ; 整数加法movl %edx, %eaxpopq %rbpret; add<double>实例化
_Z3addIdET_S0_S0_:pushq %rbpmovq %rsp, %rbpmovsd %xmm0, -8(%rbp) ; double amovsd %xmm1, -16(%rbp) ; double baddsd -16(%rbp), %xmm0 ; 浮点加法movsd %xmm0, -24(%rbp)movsd -24(%rbp), %xmm0popq %rbpretmain:; 调用add<int>movl $2, %esimovl $1, %edicall _Z3addIiET_S0_S0_; 调用add<double>movsd .LC0(%rip), %xmm1movsd .LC1(%rip), %xmm0call _Z3addIdET_S0_S0_
关键特征分析:
-
名称修饰(Name Mangling):
_Z3addIiET_S0_S0_
中的Ii
表示int
类型参数_Z3addIdET_S0_S0_
中的Id
表示double
类型参数- 不同编译器修饰规则不同(MSVC使用
??$add@H@@YAHHH@Z
格式)
-
代码生成策略:
- 即使函数逻辑相同(都是加法),
int
和double
版本仍生成独立汇编 - 每个实例化版本有独立栈帧管理(
movl
vsmovsd
指令差异)
- 即使函数逻辑相同(都是加法),
六、if constexpr
的汇编优化实证
6.1 对比实验:if
vs if constexpr
C++测试代码:
// demo_if.cpp
template<bool flag>
void test() {if constexpr (flag) { // 替换为普通if观察差异asm("nop; nop; nop"); // 插入3条空指令(标记分支1)} else {asm("nop; nop; nop; nop"); // 插入4条空指令(标记分支2)}
}int main() {test<true>();test<false>();
}
6.1.1 使用if constexpr
时的汇编输出(g++ -S -O0
):
; test<true>实例化
_ZN4testILb1EEEvv:nop; nop; nop ; 仅保留真分支代码ret; test<false>实例化
_ZN4testILb0EEEvv:nop; nop; nop; nop ; 仅保留假分支代码retmain:call _ZN4testILb1EEEvvcall _ZN4testILb0EEEvv
6.1.2 使用普通if
时的汇编输出:
; test<true>实例化
_ZN4testILb1EEEvv:cmpb $0, flag(%rip) ; 插入条件判断je .L2nop; nop; nop ; 真分支jmp .L3
.L2:nop; nop; nop; nop ; 假分支
.L3:ret; test<false>实例化的汇编逻辑类似,包含跳转指令
6.2 关键结论:
if constexpr
完全消除未采用分支的代码,生成零跳转指令- 普通
if
保留所有分支的汇编代码,增加:- 条件判断指令(
cmp
/je
) - 跳转指令(
jmp
) - 冗余代码体积(多出约30%指令)
- 条件判断指令(
七、编译器内部处理流程解析(概念图)
7.1 模板处理流程
[源代码]│▼
模板解析阶段(语法树生成)│▼
模板实例化请求(遇到具体类型)│▼
实例化上下文创建(保存模板参数)│▼
生成具体函数/类的中间表示(IR)│▼
优化阶段(内联、常量传播等)│▼
生成目标架构汇编代码
7.2 if constexpr
处理流程
[解析条件表达式]│▼
编译时求值(必须为常量表达式)│▼
若条件为真 → 编译then块,丢弃else块│
若条件为假 → 编译else块,丢弃then块│▼
生成不含条件跳转的直线代码(Straight-line Code)
八、高级应用:结合编译时分支与SIMD优化
8.1 根据类型选择SIMD指令集
template<typename T>
void simd_add(T* a, T* b, T* out, size_t n) {if constexpr (std::is_same_v<T, float>) {// 使用AVX指令集优化floatfor (size_t i = 0; i < n; i += 8) {__m256 va = _mm256_load_ps(a + i);__m256 vb = _mm256_load_ps(b + i);__m256 vc = _mm256_add_ps(va, vb);_mm256_store_ps(out + i, vc);}} else if constexpr (std::is_same_v<T, int>) {// 使用SSE4.1指令集优化intfor (size_t i = 0; i < n; i += 4) {__m128i va = _mm_load_si128((__m128i*)(a + i));__m128i vb = _mm_load_si128((__m128i*)(b + i));__m128i vc = _mm_add_epi32(va, vb);_mm_store_si128((__m128i*)(out + i), vc);}}
}
8.2 汇编对比分析
- float版本生成
vmovaps
/vaddps
等AVX指令 - int版本生成
movdqa
/paddd
等SSE指令 - 未使用的分支(如
double
处理)完全消失,避免指令集兼容性问题
九、开发者调试建议
9.1 查看模板实例化符号
# 使用nm工具查看目标文件符号
nm -C demo.o | grep "add"# 输出示例:
0000000000000000 W int add<int>(int, int)
0000000000000020 W double add<double>(double, double)
9.2 编译器诊断选项
# 打印所有模板实例化过程(Clang)
clang++ -Xclang -ast-print -fsyntax-only demo.cpp# 生成模板实例化树(GCC)
g++ -fdump-tree-original-raw demo.cpp
通过结合具体汇编示例和编译器内部流程分析,开发者可以更直观地理解模板和if constexpr
的底层行为,从而编写出既高效又可维护的现代C++代码。
相关文章:
C++17模板编程与if constexpr深度解析
一、原理深化 1.1 模板编程 1.1.1 编译器如何处理模板(补充) 模板的实例化机制存在两种模式: 隐式实例化:编译器在遇到模板具体使用时自动生成代码,可能导致多翻译单元重复实例化,增加编译时间。显式实…...
指令层级:训练大型语言模型优先处理特权指令
指令层级:训练大型语言模型优先处理特权指令 The Instruction Hierarchy: Training LLMs to Prioritize Privileged Instructions关键要点:快速浏览:1. 引言(Introduction)2. 指令层级(The Instruction Hie…...
高效人脸关键点检测算法HRNet-Facial-Landmark-Detection
高效人脸关键点检测算法HRNet-Facial-Landmark-Detection 300 W data 数据标注格式 helen/trainset/1271089376_2.jpg,2.105,642.5,643.5,453.88947199999996,562.431791,457.882374,607.338625,471.39681399999995,655.136824,489.458724,702.023327,508.36972599999996,743.…...
LeetCode 3375 题解
题解 如题所示,允许暴力,虽然是暴力,但复杂度也就O(n) 还是如昨天的题目一样,使用Set.add的方法去判断即可 分三种情况 因为是set集合的原因,所以可以排除值相同的原因 当遍历数组有值小于k就return -1 当遍历数组遇…...
leetcode_数组 189. 轮转数组
189. 轮转数组 给定一个整数数组 nums,将数组中的元素向右轮转 k 个位置,其中 k 是非负数 示例 1: 输入: nums [1,2,3,4,5,6,7], k 3输出: [5,6,7,1,2,3,4] 示例 2: 输入:nums [-1,-100,3,99], k 2输出:[3,99,-1,-100] 思…...
MySQL | 三大日志文件
Undo Log(回滚日志) 实现原理与分类 原理:Undo Log 记录的是数据修改前的旧值,通过这些旧值可以将数据恢复到修改之前的状态。它采用的是逻辑日志,即记录的是如何撤销操作,而不是物理数据的实际值。 分类…...
GIS-AI 融合引擎架构:智慧景区导览系统的毫秒级响应与千级并发优化实战
本文面向 文旅行业技术决策者、GIS 开发者、AI 算法工程师,旨在解决传统景区导览系统 定位精度低、交互体验差、运营成本高 的核心痛点,提供从技术选型到落地部署的全链路解决方案。 如需获取智慧景区导览系统解决方案请前往文章最下方获取,如…...
WSA(Windows 安卓子系统)过检测教程
windows安卓子系统WSA的root和magisk的安装教程 安卓子系统WSLWSA的rootmagisk安装 WSA(Windows 安卓子系统)过检测的方法与思路 一、引言 Windows 安卓子系统(WSA)为 Windows 用户提供了在电脑上运行安卓应用的便利。然而&…...
Spark原理及代码
一、 Spark运行架构 运行架构 Spark 框架的核心是一个计算引擎,整体来说,它采用了标准 master-slave 的结构。 Spark 框架有两个核心组件: 1、Driver Spark 驱动器节点,用于执行 Spark 任务中的 main 方法,负责实…...
esp32cam -> 服务器 | 手机 -> 服务器 直接服务器传输图片
服务器先下载python : 一、Python环境搭建(CentOS/Ubuntu通用) 一条一条执行 安装基础依赖 # CentOS sudo yum install gcc openssl-devel bzip2-devel libffi-devel zlib-devel # Ubuntu sudo apt update && sudo apt install b…...
RHCSA Linux系统 数据流和重定向 tee 命令
一.数据流和重定向 1. 数据流 (1) 标准输入(stdin,代码 0):默认从键盘获取输入,只读。 (2) 标准输出(stdout,代码 1):命令执行正确信息默认输出到屏幕,只写…...
libev实现Io复用及定时器事件服务器
客户端和服务器都绑定在了enp2s0网卡,需要SERVER_IP和SERVER_PORT改为其ip,注意不能是127.0.0.1,因为这个是lo虚拟网口。 安装libev sudo apt-get install libev-dev客户端: #include <iostream> #include <string>…...
【项目实训项目博客】prompt初版实践
通过对camel技术的理解,我们向其中添加了市场营销角色的prompt 初版设计如下: chatchainconfig.json { "chain": [ { "phase": "DemandAnalysis", "phaseType": "SimplePhase", "max_turn_step…...
底盘---全向轮(Omni Wheel)
一、基本定义与起源 定义: 全向轮是一种通过在主轮外周安装多个 垂直于主轮轴线的横向小滚轮 实现多向移动的轮式结构。小滚轮可自由转动,允许设备在纵向(主轮驱动方向)和横向(小滚轮滚动方向)同时受力&…...
Python标准库json完全指南:高效处理JSON数据
一、json库概述 JSON(JavaScript Object Notation)是一种轻量级的数据交换格式,Python的json模块提供了JSON数据的编码和解码功能。该模块可以将Python对象转换为JSON字符串(序列化),也可以将JSON字符串转换为Python对象…...
用一个实际例子快速理解MCP应用的工作步骤
已经有很多的文章介绍MCP server,MCP Client工作原理,这里不做太多介绍。但是很多介绍都只是侧重介绍概念,实际的工作原理理解起来对初学者还是不太友好。本文以一个智能旅游咨询系统为例,详细说明在利用 Model Context Protocol&…...
Element Plus 图标使用方式整理
Element Plus 图标使用方式整理 以下是 Element Plus 图标的所有使用方式,包含完整代码示例和总结表格: 1. 按需引入图标组件 适用场景:仅需少量图标时,按需导入减少打包体积 示例代码: <template><div>…...
力扣题解:142. 环形链表 II
在链表学习中,我们已经了解了单链表和双链表,两者的最后一个结点都会指向NULL;今天我们介绍的循环列表则不同,其末尾结点指向的这是链表中的一个结点。 循环链表是一种特殊类型的链表,其尾节点的指针指向头节点&#…...
图灵逆向——题七-千山鸟飞绝
目录列表 过程分析headers头部M参数分析载荷x参数分析响应数据解密分析 代码实现 一进来还是一个无限debugger,前面有讲怎么过,这里直接过掉~ 老规矩,养成习惯,先看请求头里有没有加密参数发现好像是有个M,它是个32位…...
双相机结合halcon的条码检测
以下是针对提供的C#代码的详细注释和解释,结合Halcon库的功能和代码结构进行说明: --- ### **代码整体结构** 该代码是一个基于Halcon库的条码扫描类GeneralBarcodeScan,支持单台或双台相机的条码检测,并通过回调接口返回结果。…...
Transformer Decoder Block的几个优化方案
写在前面 在大型语言模型(LLM)的演进浪潮中,Transformer 架构凭借其强大的并行计算能力和对长距离依赖的出色捕捉,奠定了核心地位。然而,标准的 Transformer Decoder Block 遵循着一种相对固定的模式:先进行自注意力(Self-Attention)捕捉上下文信息,再通过前馈神经网…...
工业科学级天文相机:跨界融合的高精密成像解决方案
随着国内科技的快速发展,工业相机领域正悄然兴起一场"天文级"的技术革命。这类兼具工业设备可靠性与天文观测精度的特殊相机,正在半导体制造、天文观测、空间探测等领域开辟新的应用疆域。其核心技术突破不仅体现在传感器性能的提升࿰…...
颠覆传统!复旦微软联合研发MagicMotion,重新定义图生视频可能性
导读简介: 尽管基于DiT的模型在生成高质量和长视频方面表现出色,但许多文本到视频的方法在精确控制物体运动和相机运动等属性方面存在不足。因此,细粒度轨迹可控的视频生成技术应运而生,这对于在现实场景中生成可控视频至关重要。…...
华为数字芯片机考2025合集5已校正
1. 题目内容 下列选项中()不是 Verilog HDL 的关键字。() A. tri B. for C. force D. edge 解析 1. Verilog 关键字分类 Verilog 关键字是语言预定义的保留字,用于语法结构或特定功能。 2. 选项分析 选项类型说明…...
QML Loader:延迟加载与动态切换
目录 引言相关阅读工程结构LoaderDelay.qml - 延迟加载实现完整代码HeavyComponent.qml代码解析运行效果 LoaderSwitch.qml - 动态切换组件完整代码代码解析运行效果 Main.qml - 主界面实现完整代码主界面结构代码解析 总结下载链接 引言 QML的Loader组件提供了一种强大的机制…...
C语言--常用的链表操作
利用C语言实现链表,并定义一些常用的操作 文章目录 链表定义新建一个链表结点打印链表插入结点头插法(常用)运行 尾插法(使用较少)运行 返回链表长度链表转置运行 合并两个有序的链表运行 删除最小结点运行 打印倒数第…...
ngx_conf_param
Ubuntu 下 nginx-1.24.0 源码分析 - ngx_conf_param-CSDN博客 定义在 src\core\ngx_conf_file.c char * ngx_conf_param(ngx_conf_t *cf) {char *rv;ngx_str_t *param;ngx_buf_t b;ngx_conf_file_t conf_file;param &cf->cycle->conf…...
C++day9
思维导图 牛客练习 练习: 将我们写的 myList 迭代器里面 operator[] 和 operator 配合异常再写一遍 #include <iostream> #include <cstring> #include <cstdlib> #include <unistd.h> #include <sstream> #include <vector>…...
算法题:两数相加
题目:2. 两数相加 给你两个 非空 的链表,表示两个非负的整数。它们每位数字都是按照 逆序 的方式存储的,并且每个节点只能存储 一位 数字。 请你将两个数相加,并以相同形式返回一个表示和的链表。 你可以假设除了数字 0 之外&a…...
SCI科学论文的重要组成部分
科学论文的核心结构 科学论文通常遵循IMRAD结构,即: 引言(Introduction)方法(Methods)结果(Results)讨论(Discussion) 除此之外,还包括其他几个关键部分。让我为您详细介绍每个部分的作用和重要性: 1. 标题(Title) 标题是论文…...
Go 微服务框架 | 路由实现
文章目录 不用框架实现web接口实现简单的路由实现分组路由支持不同的请求方式支持同一个路径的不同请求方式前缀树应用前缀树完善路由代码 不用框架实现web接口 // blog main.go 文件 package mainimport ("fmt""log""net/http" )func main() {…...
2025年AI开发学习路线
目录 一、基础阶段(2-3个月) 1. 数学与编程基础 2. 机器学习入门 二、核心技能(3-4个月) 1. 深度学习与框架 2. 大模型开发(重点) 三、进阶方向(3-6个月) 1. 多模态与智能体…...
TimescaleDB 2.19.2 发布
TimescaleDB 2.19.2 已于 2025 年 4 月 7 日发布2。此次发布是基于 PostgreSQL 的开源时序数据库 TimescaleDB 的一次更新。 从 GitHub 上的 Pull Request 信息可知,此次发布主要是将相关更改合并到 2.19.x 分支,涉及到一系列的测试和构建配置,包括不同版本 PostgreSQL(如 …...
「Unity3D」TextMeshPro中的TMP_InputField,用来实现输入框的几个小问题
第一,正确设置Scrollbar。 设置Scrollbar之后,不能设置Text Component的Font Size为Auto Size,否则Scrollbar无法正确计算显示。 那么,要想自动适配字体大小,可以让Placeholder中的Font Size设置为Auto,这…...
HTML 是什么?网页创建的核心标记语言
原文:HTML 是什么?网页创建的核心标记语言 | w3cschool笔记 HTML 是什么? HTML 是一种标记语言,用于创建网页。简单来说,HTML 就像一本魔法书,它告诉电脑如何展示网页上的内容,比如文字、图片…...
考研单词笔记 2025.04.09
act v表现,行动,做事,扮演,充当,担任,起作用n行为,行动,法案,法令 action n行为,行动 behave v表现,行事,守规矩,举止端…...
map/multimap
1.概念 map中所有元素都是pair<key,value>,key 是map的键,value 是map的值 所有元素都会根据key自动排序 map/multimap属于关联式容器,底层结构是用二叉树实现。 map和multimap区别: map不允许容器中有重复key值元素 m…...
CSS 定位属性的生动比喻:以排队为例理解 relative 与 absolute
目录 一、理解标准流与队伍的类比 二、relative 定位:队伍中 “小范围活动” 的人 三、absolute 定位:队伍中 “彻底离队” 的人 在学习 CSS 的过程中,定位属性relative和absolute常常让初学者感到困惑。它们的行为方式和对页面布局的影响较…...
基于二叉堆实现的 PriorityQueue
基于二叉堆实现的 PriorityQueue 是一种常见的数据结构,广泛用于任务调度、路径搜索、事件模拟等场景。下面我将用 Java 语言实现一个简单的基于最小堆的 PriorityQueue,即优先级最小的元素先出队。 ✅ 实现目标 使用数组实现二叉最小堆(即父…...
大模型分布式推理和量化部署
一、小常识 1、计算大模型占用多少显存 对于一个7B(70亿)参数的模型,每个参数使用16位浮点数(等于 2个 Byte)表示,则模型的权重大小约为: 7010^9 parameters2 Bytes/parameter14GB 70亿个参数…...
循环神经网络 - 长程依赖问题及改进方案
循环神经网络在学习过程中的主要问题是由于梯度消失或爆炸问题,很难建模长时间间隔(Long Range)的状态之间的依赖关系。 本文我们来学习长程依赖问题及其对应的改进方案,在这部分知识的学习过程中,我建议大家着重理解,对于数学公…...
点击抽奖功能总结
首先用户打开网页,映入眼帘的是一个输入框和一个提交按钮。当用户在输入框中输入自己的年龄并点击提交后,系统会根据输入的年龄给出相应提示。若年龄达到 60 岁,页面将显示一个新的抽奖区域,用户可以点击 “抽奖” 按钮开始抽奖。…...
AWS Bedrock生成视频详解:AI视频创作新时代已来临
💡 TL;DR: AWS Bedrock现已支持AI视频生成功能,让企业无需深厚AI专业知识即可创建高质量视频内容。本文详解Bedrock视频生成能力的工作原理、应用场景和实操指南,助你快速掌握这一革命性技术。 🎬 AWS Bedrock视频生成:改变内容创作的游戏规则 还记得几年前,制作一个专…...
理解 TOGAF®标准中的架构原则
原则是帮助组织实现其使命的基本规则和指南。它们旨在长期稳定且很少修改,在各个领域中充当决策和行动的指南针。在企业架构(EA)的背景下,原则在指导架构框架的开发和应用方面发挥着至关重要的作用。本文将探讨企业原则和架构原则…...
基于视觉密码的加密二值图像可逆数据隐藏
接下来,分享一篇论文,标题为《Multi-Party Reversible Data Hiding in Ciphertext Binary Images Based on Visual Cryptography》,由Bing Chen等人发表在《IEEE Signal Processing Letters》上。该论文提出了一种基于视觉密码学的多方可逆数…...
ubuntu22.04 中 No module named ‘_bz2‘问题解决方案
前言 本篇是介绍ubuntu22.04中 No module named ‘_bz2‘问题解决方案 网上版本很多,比如安装libbz库什么的,可能别人有用,但是我自己这边出了一堆问题 一、流程 1.1 查看bz2.xx.so文件 看自己的python版本,我新安装了个pyth…...
什么是声波,声波的传播距离受哪些因素影响?
一、声波的定义: 声波是一种机械波,它是通过介质(如空气、水、固体等)传播的振动。以下是关于声波的详细介绍: 1、声波的产生 声波是由物体的振动产生的。例如,人说话时,声带振动产生声波&…...
用PHPExcel 封装的导出方法,支持导出无限列
用PHPExcel 封装的导出方法,支持导出无限列 避免PHPExcel_Exception Invalid cell coordinate [1 异常错误 /*** EXCEL导出* param [string] $file_name 保存的文件名及表格工作区名,不加excel后缀名* param [array] $fields 二维数组* param [array] $…...
STL-stack栈和queue队列
stack栈和queue队列 在STL中 stack 和 queue 设计为容器适配器,容器适配器是使用特定容器类的封装对象作为其基础容器的类,提供一组特定的成员函数来访问其元素。 在我的STL系列中之前的容器 vector、list、deque 都是从底层类型一步步封装而来的,但是 stack 和 queue 没有…...
AI 提示词不会写?试试 PromptIDE
这段时间,AI 技术大爆炸 已经改变了我们的工作方式,而 会不会用 AI,已经成为区分工作能力的关键! 💡 在这个AI重构工作方式的时代,会用和不会用AI的人正在拉开巨大差距: √ 高手用AI——效率飙…...