【c/c++】c语言的自增操作在不同编译器的差别
示例代码
代码如下:
#include <stdio.h>#define product(x) ((x)*(x))int main(void)
{int i = 3, j, k;j = product(i++); // (i++) * (i++)k = product(++i); // (++i) * (++i)printf("%d %d\n", j, k);
}
执行结果
在Ubuntu18.04下通过GCC编译和执行的结果:
注意第一个值是12。
在Windows10下通过VS2015编译和执行的结果:
注意第一个值是9。
也就是说同样的代码,在不同的编译器下执行的结果不同!
反汇编分析
通过汇编代码分析,首先查看GCC的反汇编:
jw@ubuntu:~/code/tmp$ gcc -g test.c
jw@ubuntu:~/code/tmp$ gdb a.out
GNU gdb (Ubuntu 8.1.1-0ubuntu1) 8.1.1
Copyright (C) 2018 Free Software Foundation, Inc.
License GPLv3+: GNU GPL version 3 or later <http://gnu.org/licenses/gpl.html>
This is free software: you are free to change and redistribute it.
There is NO WARRANTY, to the extent permitted by law. Type "show copying"
and "show warranty" for details.
This GDB was configured as "x86_64-linux-gnu".
Type "show configuration" for configuration details.
For bug reporting instructions, please see:
<http://www.gnu.org/software/gdb/bugs/>.
Find the GDB manual and other documentation resources online at:
<http://www.gnu.org/software/gdb/documentation/>.
For help, type "help".
Type "apropos word" to search for commands related to "word"...
Reading symbols from a.out...done.
(gdb) l
1 #include<stdio.h>
2
3 #define product(x) ((x)*(x))
4
5 int main(void)
6 {
7 int i = 3, j, k;
8 j = product(i++); // (i++) * (i++)
9 k = product(++i); // (++i) * (++i)
10
(gdb) b 7
Breakpoint 1 at 0x652: file test.c, line 7.
(gdb) r
Starting program: /home/jw/code/tmp/a.out Breakpoint 1, main () at test.c:7
7 int i = 3, j, k;
(gdb) set disassembly-flavor intel
(gdb) disas
Dump of assembler code for function main:0x000055555555464a <+0>: push rbp0x000055555555464b <+1>: mov rbp,rsp0x000055555555464e <+4>: sub rsp,0x10
=> 0x0000555555554652 <+8>: mov DWORD PTR [rbp-0xc],0x3 ; 给i赋值为30x0000555555554659 <+15>: mov edx,DWORD PTR [rbp-0xc] ; 将i的值赋值给edx,所以edx = 30x000055555555465c <+18>: lea eax,[rdx+0x1] ; rdx的值就是edx的值,就是3,这里就是3 + 1赋值给eax,所以eax = 40x000055555555465f <+21>: mov DWORD PTR [rbp-0xc],eax ; 将eax的值赋值给i,也就是说,这里i发生了一次自增,此时i = 40x0000555555554662 <+24>: mov eax,DWORD PTR [rbp-0xc] ; 将i的值赋值给eax,所以eax = 40x0000555555554665 <+27>: lea ecx,[rax+0x1] ; rax的值就是eax的值,所以这里ecx = 50x0000555555554668 <+30>: mov DWORD PTR [rbp-0xc],ecx ; 将ecx的值赋值给i,这里又是一次i的自增,此时i = 50x000055555555466b <+33>: imul eax,edx ; 完成相乘的操作并赋值给eax,此时eax = 4, edx = 3, 所以eax = 120x000055555555466e <+36>: mov DWORD PTR [rbp-0x8],eax ; 将eax的值赋值给j,所以j = 12,到这里结果已经出来了0x0000555555554671 <+39>: add DWORD PTR [rbp-0xc],0x10x0000555555554675 <+43>: add DWORD PTR [rbp-0xc],0x10x0000555555554679 <+47>: mov eax,DWORD PTR [rbp-0xc]0x000055555555467c <+50>: imul eax,DWORD PTR [rbp-0xc]0x0000555555554680 <+54>: mov DWORD PTR [rbp-0x4],eax0x0000555555554683 <+57>: mov edx,DWORD PTR [rbp-0x4]0x0000555555554686 <+60>: mov eax,DWORD PTR [rbp-0x8]0x0000555555554689 <+63>: mov esi,eax0x000055555555468b <+65>: lea rdi,[rip+0xa2] # 0x5555555547340x0000555555554692 <+72>: mov eax,0x00x0000555555554697 <+77>: call 0x555555554520 <printf@plt>0x000055555555469c <+82>: mov eax,0x00x00005555555546a1 <+87>: leave 0x00005555555546a2 <+88>: ret
---Type <return> to continue, or q <return> to quit---
End of assembler dump.
(gdb) q
A debugging session is active.Inferior 1 [process 20205] will be killed.Quit anyway? (y or n) y
查看注释的代码,可以看到j的值是如何计算出来的。
然后分析VS的反汇编:
00E73D8E mov dword ptr [i],3 ; i赋值为3j = product(i++); // (i++) * (i++)
00E73D95 mov eax,dword ptr [i] ; eax = 3
00E73D98 imul eax,dword ptr [i] ; eax = 9
00E73D9C mov dword ptr [j],eax ; j = 9
00E73D9F mov ecx,dword ptr [i]
00E73DA2 add ecx,1
00E73DA5 mov dword ptr [i],ecx ; 完成i的一次自增
00E73DA8 mov edx,dword ptr [i]
00E73DAB add edx,1
00E73DAE mov dword ptr [i],edx ; 再完成i的一次自增k = product(++i); // (++i) * (++i)
00E73DB1 mov eax,dword ptr [i]
00E73DB4 add eax,1
00E73DB7 mov dword ptr [i],eax
00E73DBA mov ecx,dword ptr [i]
00E73DBD add ecx,1
00E73DC0 mov dword ptr [i],ecx
00E73DC3 mov edx,dword ptr [i]
00E73DC6 imul edx,dword ptr [i]
00E73DCA mov dword ptr [k],edx
相比之下VS的汇编更直观,很容易就得到了9。
通过反汇编,可以看到两边到底是如何处理这个代码的,但是也仅此而已,只能证明两个编译器通过不同的方式执行了代码,但是具体哪一种是“对”的呢?汇编代码无法告诉我们。
进一步分析
从直觉上讲,问题应该是出在i++
这个操作上,它是后置的自增操作,如果是简单的j = i++;
,则GCC和VS的执行结果是一致的。
但是如果是i = i++;
呢?将之前的代码修改:
#include <stdio.h>int main(void)
{int i = 3;i = i++;printf("%d\n", i);
}
查看GCC和VS的结果,发现确实也有差别,在GCC中的结果3,而VS中的结果是4!
从这里可以推出同一个变量在一行代码中存在执行顺序方面的不同可能,这跟编译器相关。通过查看c语言的标准(在Project status and milestones (open-std.org)可以查看c语言的标准,这里参考的是C11),里面有相关的说明:
也就是说c语言实际上将这种行为定义为undefined,所以不同的编译器就可以有不同的实现了。关于这个问题,在Stack Overflow有更多的说明,比如c - Why are these constructs using pre and post-increment undefined behavior? - Stack Overflow
总结
通常我们不会写i = i++;
这样的代码,但是当有宏定义的时候,当它扩展之后还是存在未定义行为的情况,此时不同编译器实现可能导致不同的结果。
所以在使用++
这种操作的时候需要小心,极端一点,甚至可以不用,实际上i++
和i = i + 1
可能并没有什么差异,比如在VS里面通过反汇编查看两种情况时的汇编代码:
int i = 3;
00BA1A6E mov dword ptr [i],3 i++;
00BA1A75 mov eax,dword ptr [i]
00BA1A78 add eax,1
00BA1A7B mov dword ptr [i],eax
和
int i = 3;
00C61A6E mov dword ptr [i],3 i = i + 1;
00C61A75 mov eax,dword ptr [i]
00C61A78 add eax,1
00C61A7B mov dword ptr [i],eax
实际上的汇编代码根本没有区别(这里使用的是默认的编译参数,也许使用不同的优化参数会有不同结果)。
相关文章:

【c/c++】c语言的自增操作在不同编译器的差别
示例代码 代码如下: #include <stdio.h>#define product(x) ((x)*(x))int main(void) {int i 3, j, k;j product(i); // (i) * (i)k product(i); // (i) * (i)printf("%d %d\n", j, k); }执行结果 在Ubuntu18.04下通过GCC编译和执行的结果…...

【LeetCode第 332 场周赛】
传送门 文章目录6354. 找出数组的串联值6355. 统计公平数对的数目6356. 子字符串异或查询6357. 最少得分子序列6354. 找出数组的串联值 题目 思路 前后指针 代码 class Solution { public:long long findTheArrayConcVal(vector<int>& nums) {long long res 0;i…...

【蓝桥杯单片机】Keil5中怎么添加STC头文件;从烧录软件中添加显示添加成功后新建工程时依旧找不到
蓝桥杯单片机的芯片型号:IAP15F2K61S2 添加头文件:STC15F2K60S2.H 【1】如何通过烧录软件添加STC头文件: 从ATC-ISP的Keil仿真设置中添加(同时自动下载仿真驱动)仔细阅读添加说明 KEIL5添加STC芯片库_Initdev的博客-…...

图解浏览器渲染页面详细过程
渲染详细过程 产生渲染任务,开启渲染流程 当浏览器的网络线程收到 HTML 文档后,会产生一个渲染任务,并将其传递给渲染主线程的消息队列。 在事件循环机制的作用下,渲染主线程取出消息队列中的渲染任务,开启渲染流程。…...
多线程面试题开胃菜1(5道)
一.多线程有什么用?1)发挥多核CPU 的优势随着工业的进步,现在的笔记本、台式机乃至商用的应用服务器至少也都是双核的,4 核、8 核甚至 16 核的也都不少见,如果是单线程的程序,那么在双核 CPU 上就浪费了 50…...

植物育种中广义遗传力的定义
大家好, 我是邓飞。 今天聊一下广义遗传力的计算方法。 广义遗传力定义 广义遗传力(H2H^2H2)定义为归因于基因型总体遗传变异的表型变异比例。 通常他包括三个解释:(详见我这篇博客的公式推导 回归系数 相关系数 遗…...

西瓜书读书笔记—绪论
文章目录机器学习典型的机器学习过程基本术语归纳偏好机器学习 机器学习:致力于研究如果通过计算的手段,利用经验来改善系统自身的性能 在计算机系统中,“经验” 通常以 “数据” 形式存在,因此,机器学习所研究的主要内…...

ES8——Generator函数的使用
babel工具插件下载:npm i --save babel-polyfill 引入:polyfill.js进行转码(es8->es5) 介绍 Generator函数用于生成迭代器 function * (){} yeild: 作用同return类似 {const obj function* () {yield "a";yield 12…...
德馨食品冲刺A股上市:计划募资9亿元,林志勇为实际控制人
近日,浙江德馨食品科技股份有限公司(下称“德馨食品”或“德馨饮料”)预披露更新招股书,准备在上海证券交易所主板上市。据贝多财经了解,德馨食品于2022年7月5日递交上市申请,安信证券为其保荐机构。 本次…...

湿敏电阻的原理,结构,分类与应用总结
🏡《总目录》 0,概述 湿敏电阻是指电阻值随着环境的湿度变化而变化的电阻,本文对其工作原理,结构,分类和应用场景进行总结。 1,工作原理 湿敏电阻是利用湿敏材料制成的,湿敏材料吸收空气中水分时,自身的阻值发生变化。 2,结构 如下图所示,市民电阻包括4个部分构成,…...

千锋教育嵌入式物联网教程之系统编程篇学习-03
目录 进程的终止 exit函数 _exit函数 进程退出清理 进程间的替换 进程间通信 常见通信机制 进程间通信的实质 信号 产生信号的方式 信号的默认处理方式 进程对信号的处理方式 kill函数 进程的终止 使用exit函数对进程进行终止,而return只是结束函数&a…...

升级到https
现在很多站长都会考虑将自己的站点从http升级到https,不仅是基于安全的考虑,有的也是因为第三方平台的限制,如谷歌浏览器会将http站点标记为不安全的站点,微信平台要求接入的微信小程序必须使用https等。 那如何将一个http站点升…...

【C语言】数据结构-二叉树
主页:114514的代码大冒险 qq:2188956112(欢迎小伙伴呀hi✿(。◕ᴗ◕。)✿ ) Gitee:庄嘉豪 (zhuang-jiahaoxxx) - Gitee.com 引入 我们之前已经学过线性数据结构,今天我们将介绍非线性数据结构----树 树是一种非线性的…...
c++中std::condition_variable最全用法归纳
前言 建议阅读以下文章前需先对建立 std::thread 多线程与std::mutex 锁有一定程度的熟悉 std::thread最全用法归纳 std::mutex最全用法归纳 概括 使用 std::condition_variable 的 wait 会把目前的线程 thread 停下来并且等候事件通知,而在另一个线程中可以使用…...
Python数据可视化:数据关系图表可视化
目录 1、散点图 1.1、趋势显示的二维散点图 1.2、分布显示的二维散点图 1.3、散点曲线图...
Urho3D约定
Urho3D使用以下约定和原则: 左手坐标系。正X、Y和Z轴指向右侧、上方和前方,正旋转为顺时针。度用于角度。顺时针顶点定义正面。音频音量指定为0.0(静音)到1.0(全音量)路径名使用斜杠而不是反斜杠。调用操作…...
python数据结构-列表,元组
列表 列表是Python中最通用的数据类型,可以写成方括号之间的逗号分隔值(项目)列表。 使用列表的重要事项是,列表中的项目不必是相同的类型。也就是说一个列表中的项目(元素)可以是数字,字符串,数组,字典等甚至是列表类…...

Properties类读配置文件、修改配置文件
Properties类简介(1)Properties类是专门用于读写配置文件的集合类(2)配置文件的后缀名为.properties,内容格式为:# 可以用“#”作为注释 键值 键值**注意:**键值对不需要有空格,值不需要用引号一起来。默认类型是String。键、值不可以是null(3)Properties类的方法可查找api文档…...

图解LeetCode——剑指 Offer 24. 反转链表
一、题目 定义一个函数,输入一个链表的头节点,反转该链表并输出反转后链表的头节点。 二、示例 示例: 【输入】 1->2->3->4->5->NULL 【输出】 5->4->3->2->1->NULL 限制: 0 < 节点个数 < 5000 三、…...

【C语言】“指针的运算”、“指针与数组”
文章目录一、指针运算1.指针 - 整数2.指针-指针3.指针关系运算二、指针与数组三、二级指针四、指针数组完结一、指针运算 指针可以进行整数,指针-指针,还有关系运算,其他的运算会被编译器阻止。 1.指针 - 整数 对指针进行的时候一定要注意不…...

Linux 文件类型,目录与路径,文件与目录管理
文件类型 后面的字符表示文件类型标志 普通文件:-(纯文本文件,二进制文件,数据格式文件) 如文本文件、图片、程序文件等。 目录文件:d(directory) 用来存放其他文件或子目录。 设备…...
应用升级/灾备测试时使用guarantee 闪回点迅速回退
1.场景 应用要升级,当升级失败时,数据库回退到升级前. 要测试系统,测试完成后,数据库要回退到测试前。 相对于RMAN恢复需要很长时间, 数据库闪回只需要几分钟。 2.技术实现 数据库设置 2个db_recovery参数 创建guarantee闪回点,不需要开启数据库闪回。…...

突破不可导策略的训练难题:零阶优化与强化学习的深度嵌合
强化学习(Reinforcement Learning, RL)是工业领域智能控制的重要方法。它的基本原理是将最优控制问题建模为马尔可夫决策过程,然后使用强化学习的Actor-Critic机制(中文译作“知行互动”机制),逐步迭代求解…...

Debian系统简介
目录 Debian系统介绍 Debian版本介绍 Debian软件源介绍 软件包管理工具dpkg dpkg核心指令详解 安装软件包 卸载软件包 查询软件包状态 验证软件包完整性 手动处理依赖关系 dpkg vs apt Debian系统介绍 Debian 和 Ubuntu 都是基于 Debian内核 的 Linux 发行版ÿ…...

Day131 | 灵神 | 回溯算法 | 子集型 子集
Day131 | 灵神 | 回溯算法 | 子集型 子集 78.子集 78. 子集 - 力扣(LeetCode) 思路: 笔者写过很多次这道题了,不想写题解了,大家看灵神讲解吧 回溯算法套路①子集型回溯【基础算法精讲 14】_哔哩哔哩_bilibili 完…...

解决Ubuntu22.04 VMware失败的问题 ubuntu入门之二十八
现象1 打开VMware失败 Ubuntu升级之后打开VMware上报需要安装vmmon和vmnet,点击确认后如下提示 最终上报fail 解决方法 内核升级导致,需要在新内核下重新下载编译安装 查看版本 $ vmware -v VMware Workstation 17.5.1 build-23298084$ lsb_release…...

2.Vue编写一个app
1.src中重要的组成 1.1main.ts // 引入createApp用于创建应用 import { createApp } from "vue"; // 引用App根组件 import App from ./App.vue;createApp(App).mount(#app)1.2 App.vue 其中要写三种标签 <template> <!--html--> </template>…...

ETLCloud可能遇到的问题有哪些?常见坑位解析
数据集成平台ETLCloud,主要用于支持数据的抽取(Extract)、转换(Transform)和加载(Load)过程。提供了一个简洁直观的界面,以便用户可以在不同的数据源之间轻松地进行数据迁移和转换。…...
uniapp中使用aixos 报错
问题: 在uniapp中使用aixos,运行后报如下错误: AxiosError: There is no suitable adapter to dispatch the request since : - adapter xhr is not supported by the environment - adapter http is not available in the build 解决方案&…...
力扣-35.搜索插入位置
题目描述 给定一个排序数组和一个目标值,在数组中找到目标值,并返回其索引。如果目标值不存在于数组中,返回它将会被按顺序插入的位置。 请必须使用时间复杂度为 O(log n) 的算法。 class Solution {public int searchInsert(int[] nums, …...