建议2-2:使用显式声明为signed char或unsigned char的类型来执行算术运算

在讨论本建议话题之前,我们先看看下面的这段代码的输出结果,如代码清单1-1所示。

代码清单1-1 char使用示例


#include <stdio.h>   
int main(void)  
{             
    char c=150;       
    int i=900;      
    printf("i/c=%d\n", i/c);       
    return 0;   
}

在代码清单1-1中,或许大多数人都认为它输出的结果应该是“i/c=6”,但实际的输出结果却大相径庭。前面已经讲过,char类型的变量c可以有两种类型:有符号的(signed char)和无符号的(unsigned char)。这里假设char是8位的补码字符类型,那么代码清单1-1就可能输出“i/c=-8”(signed char)或者“i/c=6”(unsigned char)两种结果。其中,在Microsoft Visual Studio 2010与GCC中的输出结果都是“i/c=-8”,如图1-4与图1-5所示。

图1-4 代码清单1-1在Microsoft Visual Studio 2010中的输出结果

图1-5 代码清单1-1在GCC中的输出结果

其实,导致这种结果最根本的原因就在于我们不能够准确地确定char类型的变量c究竟是signed char类型还是unsigned char类型。因此,我们把决策权交给编译器,而不同的编译器默认的char类型是不同的,所以最后得到的结果也就不相同。

解决这种问题的办法很简单,就是显式地将char类型的变量c声明为signed char或unsigned char类型,这样可保证结果的唯一性,如代码清单1-2所示。

代码清单1-2 unsigned char使用示例


#include <stdio.h>
int main(void)
{
    unsigned char c=150;
    int i=900;
    printf("i/c=%d\n", i/c);
    return 0;
}

这样就显式地将char类型的变量c声明为unsigned char类型,现在,后面的除法运算(i/c)与char的符号无关,所以代码清单1-2输出的结果为“i/c=6”。