[原创]C++98升级到C++20的复习旅途-从汇编及逆向角度去分析“constexpr“关键字

本文主要是介绍[原创]C++98升级到C++20的复习旅途-从汇编及逆向角度去分析“constexpr“关键字,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

[简介]
常用网名: 猪头三
出生日期: 1981.XX.XX
QQ: 643439947
个人网站: 80x86汇编小站 https://www.x86asm.org
编程生涯: 2001年~至今[共22年]
职业生涯: 20年
开发语言: C/C++、80x86ASM、PHP、Perl、Objective-C、Object Pascal、C#、Python
开发工具: Visual Studio、Delphi、XCode、Eclipse、C++ Builder
技能种类: 逆向 驱动 磁盘 文件
研发领域: Windows应用软件安全/Windows系统内核安全/Windows系统磁盘数据安全/macOS应用软件安全
项目经历: 磁盘性能优化/文件系统数据恢复/文件信息采集/敏感文件监测跟踪/网络安全检测

[序言]
最近在努力地学习C++20的相关知识点, 给自己订下一个小目标: 把自身已掌握的陈旧C++98, C++03逐步升级到C++20. 以适应现代C++开发的要求. 在学习和复习的过程中, 顺便记录疑惑点.

[新增][constexpr]
C++11引入了constexpr关键字来声明变量, 这种变量可在编译时求值并最终生成一个常量. 由于不会产生运行时开销, 所以编译能执行额外的优化来提高应用程序的性能.

[什么是"编译时", 什么是"运行时"]
要了解"constexpr"的作用前提是, 一定要弄清楚两个概念细节"编译时"和"运行时".

"编译时": 分析和解析源代码文件的过程, 比如语法检查, 词法分析, 优化代码...
"运行时": 程序的运行过程中

理解这个两个概念之后就很好理解下面的代码了.

比如想要一个求平方函数constexpr_fun_Square()在"编译时"就运行起来, 那么就需要在函数前面添加"constexpr"关键字

// 编译时执行函数 (求平方)
constexpr int constexpr_fun_Square(int int_param_X) {return  int_param_X * int_param_X ;
}

比如想要一个求平方函数fun_Square()在程序启动之后才能执行, 那么就按照正常的函数声明即可.

// 运行时执行函数 (求平方)
int fun_Square(int int_param_X) {return  int_param_X * int_param_X ;
}

[在代码中适当的引用"编译时"代码, 为什么会提升应用程序的性能呢?]
要理解这个核心, 可以通过逆向分析, 观察"编译时"代码和"运行时"代码的差异.

1> 首先启动相关的C/C++的开发工具(我使用的是 C++ Builder 12), 创建一个C++命令控制台程序, 把如下代码整合到新建的项目中, 然后编译运行.

#include <iostream>
#include <locale>// 编译时执行函数 (求平方)
constexpr int constexpr_fun_Square(int int_param_X) {return  int_param_X * int_param_X ;
}// 运行时执行函数 (求平方)
int fun_Square(int int_param_X) {return  int_param_X * int_param_X ;
}int _tmain(int argc, _TCHAR* argv[])
{// 1> 把编译时结果赋值给编译时变量constexpr int int_Square_A = constexpr_fun_Square(5) ;// 2> 把运行时结果赋值给运行时变量int int_Square_C = fun_Square(5);// 3> 把编译时结果赋值给运行时变量int int_Square_B = constexpr_fun_Square(5) ;}

2> 对"constexpr int int_Square_A = constexpr_fun_Square(5) ;"行下断点之后, 以Debug模式启动运行
3> 当程序被断下来之后, 就切换到汇编指令模式, 得到如下汇编代码.

File1.cpp.15: int _tmain(int argc, _TCHAR* argv[])
005814F0 55               push ebp
005814F1 89E5             mov ebp,esp
005814F3 83EC28           sub esp,$28
005814F6 8B450C           mov eax,[ebp+$0c]
005814F9 8B4D08           mov ecx,[ebp+$08]
005814FC BA05000000       mov edx,$00000005
00581501 C745FC00000000   mov [ebp-$04],$00000000
File1.cpp.18: constexpr int int_Square_A = constexpr_fun_Square(5) ;
00581508 C745F819000000   mov [ebp-$08],$00000019 // 1> "编译时"得到了优化
File1.cpp.21: int int_Square_C = fun_Square(5);
0058150F C7042405000000   mov [esp],$00000005
00581516 8945EC           mov [ebp-$14],eax
00581519 894DE8           mov [ebp-$18],ecx
0058151C 8955E4           mov [ebp-$1c],edx
0058151F E8ACFFFFFF       call fun_Square(int)  // 2> 正常调用, 因为fun_Square()函数是运行时执行
00581524 B905000000       mov ecx,$00000005
00581529 8945F4           mov [ebp-$0c],eax
File1.cpp.24: int int_Square_B = constexpr_fun_Square(5) ;
0058152C C7042405000000   mov [esp],$00000005
00581533 894DE0           mov [ebp-$20],ecx
00581536 E80D000000       call constexpr_fun_Square(int) // 3> "编译时"没有优化
0058153B 8945F0           mov [ebp-$10],eax
File1.cpp.26: }

通过观察如上的汇编代码, 惊奇的发现 "constexpr int int_Square_A = constexpr_fun_Square(5) ;" 这段代码并没有调用constexpr_fun_Square()函数, 而是直接赋值, 效果相当于如下写法:

constexpr int int_Square_A = constexpr_fun_Square(5) ;
等价于
const int int_Square_A = 25;
且等价于汇编代码
00581508 C745F819000000   mov [ebp-$08],$00000019

这意味着什么?意味着这个程序运行的时候少了调用constexpr_fun_Square(5) 的环节, 那继续意味着什么? 就是大大提升了程序的运行效率.

[不要开心, 下面一个重要的细节: 3> 把编译时结果赋值给运行时变量]
当程序如果运行到如下代码, 又会发生什么情况?:

// 3> 把编译时结果赋值给运行时变量
int int_Square_B = constexpr_fun_Square(5) ;

找到并观察对应的汇编代码

0058152C C7042405000000   mov [esp],$00000005
00581533 894DE0           mov [ebp-$20],ecx
00581536 E80D000000       call constexpr_fun_Square(int)

结果发现, 不是想象中那么美好,  程序调用(call) constexpr_fun_Square(int)这个函数, 没有提升运行效率,为什么会这样呢?这是因为int_Square_B变量并不是constexpr变量, 因此编译器没有针对它进行"编译时"优化.

[结尾]
这是一个全新的角度来分析和理解constexpr关键字的作用, 只有真正通过逆向观察, 才能有更深地体会, 更容易理解书本上的文字描述. 希望大家喜欢这篇文章, 如果有对文章有更多的疑问, 可以留言, 我会一一认真回复的.

[截图欣赏]

这篇关于[原创]C++98升级到C++20的复习旅途-从汇编及逆向角度去分析“constexpr“关键字的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/450918

相关文章

C++使用栈实现括号匹配的代码详解

《C++使用栈实现括号匹配的代码详解》在编程中,括号匹配是一个常见问题,尤其是在处理数学表达式、编译器解析等任务时,栈是一种非常适合处理此类问题的数据结构,能够精确地管理括号的匹配问题,本文将通过C+... 目录引言问题描述代码讲解代码解析栈的状态表示测试总结引言在编程中,括号匹配是一个常见问题,尤其是在

使用C++实现链表元素的反转

《使用C++实现链表元素的反转》反转链表是链表操作中一个经典的问题,也是面试中常见的考题,本文将从思路到实现一步步地讲解如何实现链表的反转,帮助初学者理解这一操作,我们将使用C++代码演示具体实现,同... 目录问题定义思路分析代码实现带头节点的链表代码讲解其他实现方式时间和空间复杂度分析总结问题定义给定

C++初始化数组的几种常见方法(简单易懂)

《C++初始化数组的几种常见方法(简单易懂)》本文介绍了C++中数组的初始化方法,包括一维数组和二维数组的初始化,以及用new动态初始化数组,在C++11及以上版本中,还提供了使用std::array... 目录1、初始化一维数组1.1、使用列表初始化(推荐方式)1.2、初始化部分列表1.3、使用std::

C++ Primer 多维数组的使用

《C++Primer多维数组的使用》本文主要介绍了多维数组在C++语言中的定义、初始化、下标引用以及使用范围for语句处理多维数组的方法,具有一定的参考价值,感兴趣的可以了解一下... 目录多维数组多维数组的初始化多维数组的下标引用使用范围for语句处理多维数组指针和多维数组多维数组严格来说,C++语言没

Springboot中分析SQL性能的两种方式详解

《Springboot中分析SQL性能的两种方式详解》文章介绍了SQL性能分析的两种方式:MyBatis-Plus性能分析插件和p6spy框架,MyBatis-Plus插件配置简单,适用于开发和测试环... 目录SQL性能分析的两种方式:功能介绍实现方式:实现步骤:SQL性能分析的两种方式:功能介绍记录

最长公共子序列问题的深度分析与Java实现方式

《最长公共子序列问题的深度分析与Java实现方式》本文详细介绍了最长公共子序列(LCS)问题,包括其概念、暴力解法、动态规划解法,并提供了Java代码实现,暴力解法虽然简单,但在大数据处理中效率较低,... 目录最长公共子序列问题概述问题理解与示例分析暴力解法思路与示例代码动态规划解法DP 表的构建与意义动

c++中std::placeholders的使用方法

《c++中std::placeholders的使用方法》std::placeholders是C++标准库中的一个工具,用于在函数对象绑定时创建占位符,本文就来详细的介绍一下,具有一定的参考价值,感兴... 目录1. 基本概念2. 使用场景3. 示例示例 1:部分参数绑定示例 2:参数重排序4. 注意事项5.

使用C++将处理后的信号保存为PNG和TIFF格式

《使用C++将处理后的信号保存为PNG和TIFF格式》在信号处理领域,我们常常需要将处理结果以图像的形式保存下来,方便后续分析和展示,C++提供了多种库来处理图像数据,本文将介绍如何使用stb_ima... 目录1. PNG格式保存使用stb_imagephp_write库1.1 安装和包含库1.2 代码解

C++实现封装的顺序表的操作与实践

《C++实现封装的顺序表的操作与实践》在程序设计中,顺序表是一种常见的线性数据结构,通常用于存储具有固定顺序的元素,与链表不同,顺序表中的元素是连续存储的,因此访问速度较快,但插入和删除操作的效率可能... 目录一、顺序表的基本概念二、顺序表类的设计1. 顺序表类的成员变量2. 构造函数和析构函数三、顺序表

使用C++实现单链表的操作与实践

《使用C++实现单链表的操作与实践》在程序设计中,链表是一种常见的数据结构,特别是在动态数据管理、频繁插入和删除元素的场景中,链表相比于数组,具有更高的灵活性和高效性,尤其是在需要频繁修改数据结构的应... 目录一、单链表的基本概念二、单链表类的设计1. 节点的定义2. 链表的类定义三、单链表的操作实现四、