【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.指针 - 整数 对指针进行的时候一定要注意不…...

IDEA运行Tomcat出现乱码问题解决汇总
最近正值期末周,有很多同学在写期末Java web作业时,运行tomcat出现乱码问题,经过多次解决与研究,我做了如下整理: 原因: IDEA本身编码与tomcat的编码与Windows编码不同导致,Windows 系统控制台…...

微信小程序之bind和catch
这两个呢,都是绑定事件用的,具体使用有些小区别。 官方文档: 事件冒泡处理不同 bind:绑定的事件会向上冒泡,即触发当前组件的事件后,还会继续触发父组件的相同事件。例如,有一个子视图绑定了b…...
深入浅出:JavaScript 中的 `window.crypto.getRandomValues()` 方法
深入浅出:JavaScript 中的 window.crypto.getRandomValues() 方法 在现代 Web 开发中,随机数的生成看似简单,却隐藏着许多玄机。无论是生成密码、加密密钥,还是创建安全令牌,随机数的质量直接关系到系统的安全性。Jav…...

USB Over IP专用硬件的5个特点
USB over IP技术通过将USB协议数据封装在标准TCP/IP网络数据包中,从根本上改变了USB连接。这允许客户端通过局域网或广域网远程访问和控制物理连接到服务器的USB设备(如专用硬件设备),从而消除了直接物理连接的需要。USB over IP的…...

20个超级好用的 CSS 动画库
分享 20 个最佳 CSS 动画库。 它们中的大多数将生成纯 CSS 代码,而不需要任何外部库。 1.Animate.css 一个开箱即用型的跨浏览器动画库,可供你在项目中使用。 2.Magic Animations CSS3 一组简单的动画,可以包含在你的网页或应用项目中。 3.An…...
【SpringBoot自动化部署】
SpringBoot自动化部署方法 使用Jenkins进行持续集成与部署 Jenkins是最常用的自动化部署工具之一,能够实现代码拉取、构建、测试和部署的全流程自动化。 配置Jenkins任务时,需要添加Git仓库地址和凭证,设置构建触发器(如GitHub…...

nnUNet V2修改网络——暴力替换网络为UNet++
更换前,要用nnUNet V2跑通所用数据集,证明nnUNet V2、数据集、运行环境等没有问题 阅读nnU-Net V2 的 U-Net结构,初步了解要修改的网络,知己知彼,修改起来才能游刃有余。 U-Net存在两个局限,一是网络的最佳深度因应用场景而异,这取决于任务的难度和可用于训练的标注数…...
MFE(微前端) Module Federation:Webpack.config.js文件中每个属性的含义解释
以Module Federation 插件详为例,Webpack.config.js它可能的配置和含义如下: 前言 Module Federation 的Webpack.config.js核心配置包括: name filename(定义应用标识) remotes(引用远程模块࿰…...

热门Chrome扩展程序存在明文传输风险,用户隐私安全受威胁
赛门铁克威胁猎手团队最新报告披露,数款拥有数百万活跃用户的Chrome扩展程序正在通过未加密的HTTP连接静默泄露用户敏感数据,严重威胁用户隐私安全。 知名扩展程序存在明文传输风险 尽管宣称提供安全浏览、数据分析或便捷界面等功能,但SEMR…...

《信号与系统》第 6 章 信号与系统的时域和频域特性
目录 6.0 引言 6.1 傅里叶变换的模和相位表示 6.2 线性时不变系统频率响应的模和相位表示 6.2.1 线性与非线性相位 6.2.2 群时延 6.2.3 对数模和相位图 6.3 理想频率选择性滤波器的时域特性 6.4 非理想滤波器的时域和频域特性讨论 6.5 一阶与二阶连续时间系统 6.5.1 …...