由于临时参加了另一个比赛,无缘今年省赛,但是还是去333看了一下,和S大佬解决了一个C语言中运算问题。当时问题是解决了,现在详细分析一下。
先描述一下这个问题:
Z同学的代码中,这样写到(意思相近):
unsigned int numA;
int numB;
result = (numB-numA)/10;
当 numA - numB 为负数时,给出的不是负数,而是一个巨大的正整数(毫无疑问把负数补码当作正整数了)
S大佬给出的方案是:(当时大家还没注意到一个unsigned的变量)
unsigned int numA;
int numB,temp;
temp = numB-numA;
result = temp/10;
成功解决了问题。
最后注意到这个 unsigned的变量 ,于是修改成
int numA,numB;
result = (numB-numA)/10;
彻底解决了问题,原因是在最初整体计算中,(numB-numA)的结果是一个unsigned int型数值,
C语言将这个数值最高位视为数据而非符号,因此导致问题的产生。这个问题是隐式转换的锅:
(介绍来自https://blog.csdn.net/u011240016/article/details/52567864)
1,隐式转换
C在以下四种情况下会进行隐式转换:
1、算术运算式中,低类型能够转换为高类型。
2、赋值表达式中,右边表达式的值自动隐式转换为左边变量的类型,并赋值给他。
3、函数调用中参数传递时,系统隐式地将实参转换为形参的类型后,赋给形参。
4、函数有返回值时,系统将隐式地将返回表达式类型转换为返回值类型,赋值给调用函数。
2,算数运算的隐式转换
算数运算中,首先有如下类型转换规则:
1、字符必须先转换为整数(C语言规定字符类型数据和整型数据之间可以通用) 。
2、short型转换为int型(同属于整型) 。
3、float型数据在运算时一律转换为双精度(double)型,以提高运算精度(同属于实型) 。
其次,有下面的规则。
当不同类型的数据进行操作时,应当首先将其转换成相同的数据类型,然后进行操作,转换规则是由低级向高级转换。转换规则如下图所示:
从图中可知 unsigned 比 int 高级,因此在数值计算发生了一个int 到 unsigned int的转换(简直是一个教科书级别的错误实例)。