[原创]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关键字的作用, 只有真正通过逆向观察, 才能有更深地体会, 更容易理解书本上的文字描述. 希望大家喜欢这篇文章, 如果有对文章有更多的疑问, 可以留言, 我会一一认真回复的.

[截图欣赏]

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/213052.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

AtCoder Beginner Contest 331 题解 A-E

目录 A - TomorrowB - Buy One Carton of MilkC - Sum of Numbers Greater Than MeD - Tile PatternE - Set Meal A - Tomorrow 原题链接 题目描述 已知一年有M个月D天&#xff0c;求出第y年m月d天的后一天是哪一天。 思路&#xff1a;分类讨论 分别讨论m和d的是否是最后一个月…

基于SpringBoot的旅游信息网【源码好优多】

简介 旅游信息网是一款介绍旅游相关内容的网站&#xff0c;分为前台和后台部分&#xff0c;其中前台用户注册以后可以浏览景点、景点详情、预订景点、酒店、车票、保险、以及浏览旅游攻略、个人信息修改、在线留言等&#xff0c;管理员在后台对景点、攻略、订单信息、酒店信息、…

oj赛氪练习题

数组调整 import java.util.Scanner;public class Main {public static void main(String[] args) {Scanner scanner new Scanner(System.in);int n scanner.nextInt();int k scanner.nextInt();int[] arr new int[n];for (int i 0; i < n; i) {arr[i] scanner.nextIn…

java源码-类与对象

1、面向对象与面向过程 在了解类和对象之前我们先了解一下什么是面向过程和面向对象。 1&#xff09;面向过程编程&#xff1a; C语言就是面向过程编程的&#xff0c;关注的是过程&#xff0c;分析出求解问题的步骤&#xff0c;通过函数调用逐步解决问题。 2&#xff09;面向对…

Redis 发布订阅机制深入探索

Redis 的发布订阅&#xff08;pub/sub&#xff09;机制是一种消息传递模式&#xff0c;允许消息的发送者&#xff08;发布者&#xff09;和消息的接收者&#xff08;订阅者&#xff09;通过一个中介层&#xff08;频道&#xff09;进行通信&#xff0c;而无需彼此直接交互。以下…

半导体工艺发展概述

集成电路发展到今天&#xff0c;经历从1940年的PN结发现&#xff0c;到1950年BJT三极管发明&#xff0c;再到1963年CMOS电路发明。从单纯基于Si的半导体电路&#xff0c;再到GaAs, GaN&#xff0c;SiGe, InP等化合物半导体集成电路。不断的通过化学材料配比&#xff0c;基本单元…

TinyVue 组件库助力赛意信息获得工业软件种子奖

首先恭喜广州赛意信息科技股份有限公司荣获工业软件种子奖&#xff01;在本次大赛中&#xff0c;凭借“数据驱动智造&#xff0c;基于 iDME 的赛意新一代 SMOM 赋能电子行业制造运营管理解决方案”这一作品脱颖而出~ 大赛简介 10月30日至10月31日&#xff0c;由广东省工业和信…

圆通速递查询入口,以表格的形式导出单号的每一条物流信息

批量查询圆通速递单号的物流信息&#xff0c;以表格的形式导出单号的每一条物流信息。 所需工具&#xff1a; 一个【快递批量查询高手】软件 圆通速递单号若干 操作步骤&#xff1a; 步骤1&#xff1a;运行【快递批量查询高手】软件&#xff0c;并登录 步骤2&#xff1a;点击…

Hadoop——分布式计算MapReduce和资源调度Yarn

分布式计算 MapReduce YARN架构 YARN集群部署 一、Hadoop安装目录下/etc/hadoop修改mapred-env配置文件&#xff0c;mapred-site.xml文件 二、etc/hadoop文件内&#xff0c;修改yarn-env.sh&#xff0c;yarn-site.xml 三、将配置好的文件分发到其他服务节点 start-dfs.…

SLAM ORB-SLAM2(10)轨迹跟踪过程

SLAM ORB-SLAM2(10)轨迹跟踪过程 1. 总体过程2. ORB 特征点提取2.1. 相机数据处理2.1.1. 单目相机图像处理2.1.2. 双目相机图像处理2.1.3. RGBD相机图像处理2.2. ORB 特征点3. 地图初始化3.1. 坐标形式3.2. 坐标原点3.3. 地图尺度4. 相机位姿初始估计4.1. 关键帧4.2. 运动模型…

文件搜索神器—Everything,结合内网穿透秒变在线搜索神器!

Everythingcpolar搭建在线资料库&#xff0c;实现随时随地访问 文章目录 Everythingcpolar搭建在线资料库&#xff0c;实现随时随地访问前言1.软件安装完成后&#xff0c;打开Everything2.登录cpolar官网 设置空白数据隧道3.将空白数据隧道与本地Everything软件结合起来总结 前…

【每日一题】1423. 可获得的最大点数-2023.12.3

题目&#xff1a; 1423. 可获得的最大点数 几张卡牌 排成一行&#xff0c;每张卡牌都有一个对应的点数。点数由整数数组 cardPoints 给出。 每次行动&#xff0c;你可以从行的开头或者末尾拿一张卡牌&#xff0c;最终你必须正好拿 k 张卡牌。 你的点数就是你拿到手中的所有…

【Android】解决安卓中并不存在ActivityMainBinding

安卓中并不存在ActivityMainBinding这个类&#xff0c;这个类是在XML布局的最外层加入就会自动生成。但是你在最后绑定主布局时会报错获取不到根节点getRoot(). 最好的办法就是&#xff0c;删除原来的最外层节点&#xff0c;再重新添加&#xff0c;感觉是因为复制时并没有让系…

[蓝桥杯 2020 省 AB1] 解码

做题前思路&#xff1a; 1.因为是多组输入&#xff0c;又包含字符于是我们可以先定义一个char类型数组arr 2.定义数组的长度&#xff1a;题目说简写&#xff08;字母加数字&#xff09;长度不超过100&#xff0c;但原来的长度可能超过100&#xff0c;加上小明不会将连续超过9…

Pandas时序数据分析实践—基础(1)

目录 1. Pandas基本结构2. Pandas数据类型2.1. 类型概述2.1.1. 整数类型&#xff08;int&#xff09;&#xff1a;2.1.2. 浮点数类型&#xff08;float&#xff09;&#xff1a;2.1.3. 布尔类型&#xff08;bool&#xff09;&#xff1a;2.1.4. 字符串类型&#xff08;object&a…

树_对称二叉树

//给你一个二叉树的根节点 root &#xff0c; 检查它是否轴对称。 // // // // 示例 1&#xff1a; // // //输入&#xff1a;root [1,2,2,3,4,4,3] //输出&#xff1a;true // // // 示例 2&#xff1a; // // //输入&#xff1a;root [1,2,2,null,3,null,3] //输出…

JVM内存结构:StringTable与常量池关系

首先看一道题 这就涉及到StringTable和常量池&#xff0c;答案在文末&#xff0c;全做对就不用看了 而StringTable的位置在不同版本也有变化 &#xff0c; 我们只探讨jdk1.8版本 与StringTable 串池对应的是常量池 案例一、常量池和串池联系 引用所指肯定不会是常量池中的字…

vue3中如何实现事件总线eventBus

使用插件 由于vue3中 “$ on”&#xff0c;$ off 和 $ once 实例方法已被移除&#xff0c;组件实例不再实现事件触发接口 所以我们可以使用官方推荐的这个第三方库实现同样的效果 mitt https://github.com/developit/mitt 安装 pnpm install mitt -S挂载全局写法 main.ts 初始…

机械专业个人简历17篇

以下简历内容以机械专业相关岗位招聘需求为背景&#xff0c;我们整理了17篇且具有参考价值的简历案例&#xff0c;大家可以灵活借鉴&#xff0c;助理大家在众多候选人中脱颖而出。 机械专业简历模板下载&#xff08;可在线编辑制作&#xff09;&#xff1a;来幻主简历&#xf…

使用python streamlit库快速创建一个购物网站

streamlit Streamlit 是一个基于 Python 的 Web 应用程序框架&#xff0c;致力于以更高效、更灵活的方式可视化数据&#xff0c;并分析结果。 Streamlit是一个开源库&#xff0c;可以帮助数据科学家和学者在短时间内开发机器学习 (ML) 可视化仪表板。只需几行代码&#xff0c…