当前位置: 首页 > news >正文

15.75.【C语言】表达式求值

目录

一.整型提升

1.定义

2.

一.整型提升

1.定义

C语言中整型算术运算总是至少以缺省(默认)整型类型的精度来进行的。为了获得这个精度,表达式中的字符和短整型操作数在使用之前被转换为普通整型,这种转换称为整型提升

2.整型提升的原因:

表达式的整型运算要在CPU的相应运算器件内执行,CPU内整型运算器(ALU)的操作数的字节长度一般就是int的字节长度,同时也是CPU的通用寄存器的长度。
因此,即使两个char类型的相加,在CPU执行时实际上也要先转换为CPU内整型操作数的标准长度
通用CPU(general-purpose CPU)是难以直接实现两个8比特字节直接相加运算(虽然机器指令中可能有这种字节相加指令)。所以,表达式中各种长度可能小于int长度的整型值,都必须先转换为int或unsigned int,然后才能送入CPU去执行运算。

char a;//char本质上是存储的是ASCII值 
short b;

char,short-->int 或 unsigned int

3.方法:

*有符号整数提按照变量的数据类型的符号位来提升至int的位数

char a;-->signed char a;

+5原码=反码=补码=00000000 00000000 00000000 00000101 (作int时)(从8bit-->32bit)

由于char只能存8 bit,所以变量里存储00000000 00000000 00000000 00000101,称为整型截断:一个字节大的整型数据赋值给一个字节小的整型变量时,发生的数据丢失

*无符号整数提升,高位补0至int的位数

 练习:求打印的值

#include <stdio.h>
int main()
{char a=5;char b=126;char c=a+b;printf("%d",c);
}

分析:计算(a+b)前char整型提升至int

5:      0000000 00000000 00000000 00000101

126:    00000000 00000000 00000000 01111110

5+126: 00000000 00000000 0000000 10000011

存储至c时,整型截断:00000000 00000000 0000000 100000011

所以c的二进制序列为100000011,c的十进制序列为131,但打印的结果不是131

要充分理解:字符和短整型操作数使用之前被转换为普通整型的含义,printf也算使用!

解释1:

VS2022中char默认按signed char处理,最高位是1,所以高位补1

补码:11111111 11111111 11111111 10000011

符号位不变,其余各位取反再+1:10000000 00000000 00000000 01111101-->-125

解释2:“环绕溢出”

由于8bit存储有符号整数范围-128~+127

所以画图:


练习:代码修改后求打印的值

#include <stdio.h>
int main()
{unsigned char a = 5;unsigned char b = 126;unsigned char c = a + b;printf("%d", c);
}

分析指定无符号(unsigned char)整数,范围是0~2的8次方-1即255,显然131<255,所以打印131

二.算术转换

1.定义

如果某个操作符的各个操作数属于不同的类型,那么除非其中一个操作数的转换为另一个操作数的类型,否则操作就无法进行

2.剖析

float a;
double b;
a+b;

a+b,a是float,b是double,两者类型不一样,VS会把float转换为double,执行a+b;具体原因参见第3点

3.寻常算术转换

1. long double
2. double
3. float
4. unsigned long int
5. long int
6. unsigned int
7. int

规则:序号大的转换为序号小的

序号3是float,序号2是double,2<3,所以把float转换为double

4.问题表达式例子分析

表达式的执行看优先级(回顾15.25【C语言】操作符的属性),而且表达式真正计算的时候先看相邻操作符的优先级再决定先算谁,但有了优先级就一定能确定唯一的运算顺序吗?

a*b + c*d + e*f

执行顺序有两种可能:

1.先算完*,后算完+

2.a*b-->c*d-->a*b + c*d-->e*f-->a*b + c*d + e*f

显然不能确定唯一的运算顺序

注意:不同的运算顺序可能答案有所不同,上述a,b,c,d,e,f可以是变量,也可以是表达式


int c=3;
int b=c + --c;

查优先级可知,先--后+ ,但+号两边都含c,+的左操作数在--c之前还是之后准备好的,无从得知,建议再设一个变量


出自《C和指针》

int main()
{int i = 10;i = i-- - --i * ( i = -3 ) * i++ + ++i;printf("i = %d\n", i);return 0;
}

理由和上方解释思想一样 ,不同编译器解释出来的答案不一样


下列代码的输出结果是否可求?

#include <stdio.h>
int fun()
{static int count = 1;return ++count;
}int main()
{int answer=0;answer = fun() - fun() * fun();printf( "%d\n", answer);return 0;
}

注意到 static int 回顾17.【C语言】初识常见关键字 下

摘取:

int a=1;等价于auto int a=1;

static int a=1;修饰局部变量,a不会自动销毁,生命周期变长

总结:static修饰局部变量时改变了变量的存储类型,进而改变了局部变量的生命周期

count会从1到4而且函数的调用先后顺序无法通过操作符的优先级确定


#include <stdio.h>
int main()
{int i = 1;int result = (++i) + (++i) + (++i);printf("%d\n", result);printf("%d\n", i);return 0;
}

Debug x86环境下编译 ,反汇编

#include <stdio.h>
int main()
{
006B1870  push        ebp  
006B1871  mov         ebp,esp  
006B1873  sub         esp,0D8h  
006B1879  push        ebx  
006B187A  push        esi  
006B187B  push        edi  
006B187C  lea         edi,[ebp-18h]  
006B187F  mov         ecx,6  
006B1884  mov         eax,0CCCCCCCCh  
006B1889  rep stos    dword ptr es:[edi]  
006B188B  mov         ecx,offset _2D923C74_FileName@c (06BC008h)  
006B1890  call        @__CheckForDebuggerJustMyCode@4 (06B132Ah)  
006B1895  nop  int i = 1;
006B1896  mov         dword ptr [i],1  int result = (++i) + (++i) + (++i);
006B189D  mov         eax,dword ptr [i]  
006B18A0  add         eax,1  
006B18A3  mov         dword ptr [i],eax  
006B18A6  mov         ecx,dword ptr [i]  
006B18A9  add         ecx,1  
006B18AC  mov         dword ptr [i],ecx  
006B18AF  mov         edx,dword ptr [i]  
006B18B2  add         edx,1  
006B18B5  mov         dword ptr [i],edx  
006B18B8  mov         eax,dword ptr [i]  
006B18BB  add         eax,dword ptr [i]  
006B18BE  add         eax,dword ptr [i]  
006B18C1  mov         dword ptr [result],eax  printf("%d\n", result);
006B18C4  mov         eax,dword ptr [result]  
006B18C7  push        eax  
006B18C8  push        offset string "%d\n" (06B7B30h)  
006B18CD  call        _printf (06B10D2h)  
006B18D2  add         esp,8  printf("%d\n", i);
006B18D5  mov         eax,dword ptr [i]  
006B18D8  push        eax  
006B18D9  push        offset string "%d\n" (06B7B30h)  
006B18DE  call        _printf (06B10D2h)  
006B18E3  add         esp,8  return 0;
006B18E6  xor         eax,eax  
}
006B18E8  pop         edi  
006B18E9  pop         esi  
006B18EA  pop         ebx  
006B18EB  add         esp,0D8h  
006B18F1  cmp         ebp,esp  
006B18F3  call        __RTC_CheckEsp (06B124Eh)  
006B18F8  mov         esp,ebp  
006B18FA  pop         ebp  
006B18FB  ret  

int i = 1;以上的汇编指令是栈区的初始化,具体分析见36.【C语言】函数栈帧的创建和销毁

重点分析int i=1;和int result这行代码

总结:即使有了操作符的优先级和结合性,我们写出的表达式依然有可能不能通过操作符的属性确定唯一的计算路径,那这个表达式就是存在潜在风险的,建议不要写出特别复杂的表达式

相关文章:

  • 北京网站建设多少钱?
  • 辽宁网页制作哪家好_网站建设
  • 高端品牌网站建设_汉中网站制作
  • Dubbo源码深度解析(中)
  • yum 方式下载安装 java 1.8
  • Android SurfaceFlinger——渲染开始帧(四十三)
  • MySQL基础练习题22-第二高的薪水
  • C#:通用方法总结—第15集
  • AGI思考探究的意义、价值与乐趣Ⅳ
  • 36k Star的开源大模型应用开发平台,太强了!
  • cdlinux虚拟机iso文件
  • Leetcode梦开始的地方--两数相加
  • 聊聊跨境电商平台与固定IP的那些事
  • ECMAScript 12 (ES12, ES2021) 新特性
  • C:关于static 和 extern 关键字的介绍-学习笔记
  • electron-updater实现electron全量更新和增量更新——渲染进程UI部分
  • 设计模式 之 —— 抽象工厂模式
  • C++生化危机1.5源码
  • 《Java编程思想》读书笔记-对象导论
  • 「前端早读君006」移动开发必备:那些玩转H5的小技巧
  • 【140天】尚学堂高淇Java300集视频精华笔记(86-87)
  • 2017 年终总结 —— 在路上
  • 8年软件测试工程师感悟——写给还在迷茫中的朋友
  • C++类中的特殊成员函数
  • CentOS从零开始部署Nodejs项目
  • ECMAScript6(0):ES6简明参考手册
  • Java教程_软件开发基础
  • Js基础知识(四) - js运行原理与机制
  • leetcode378. Kth Smallest Element in a Sorted Matrix
  • maya建模与骨骼动画快速实现人工鱼
  • October CMS - 快速入门 9 Images And Galleries
  • open-falcon 开发笔记(一):从零开始搭建虚拟服务器和监测环境
  • React系列之 Redux 架构模式
  • 翻译 | 老司机带你秒懂内存管理 - 第一部(共三部)
  • 给Prometheus造假数据的方法
  • 构建工具 - 收藏集 - 掘金
  • 机器学习学习笔记一
  • 检测对象或数组
  • 每天10道Java面试题,跟我走,offer有!
  • 普通函数和构造函数的区别
  • 驱动程序原理
  • 数组大概知多少
  • 详解移动APP与web APP的区别
  • 移动端 h5开发相关内容总结(三)
  • 由插件封装引出的一丢丢思考
  • 在Mac OS X上安装 Ruby运行环境
  • 【运维趟坑回忆录】vpc迁移 - 吃螃蟹之路
  • 我们雇佣了一只大猴子...
  • # Redis 入门到精通(八)-- 服务器配置-redis.conf配置与高级数据类型
  • #if和#ifdef区别
  • #在 README.md 中生成项目目录结构
  • #中国IT界的第一本漂流日记 传递IT正能量# 【分享得“IT漂友”勋章】
  • $Django python中使用redis, django中使用(封装了),redis开启事务(管道)
  • (C#)一个最简单的链表类
  • (cos^2 X)的定积分,求积分 ∫sin^2(x) dx
  • (LNMP) How To Install Linux, nginx, MySQL, PHP
  • (SpringBoot)第七章:SpringBoot日志文件
  • (vue)el-cascader级联选择器按勾选的顺序传值,摆脱层级约束