环境:centos 8.4 gcc 8.4 x86_64
今天遇到一个出乎之前预期的行为,
#include <stdio.h>
int main(int argc, char *argv[])
{
char a = 0xe6;
printf("a:%02x\n", a);
return 0;
}
预期输出:
a:e6
实际输出:
a:ffffffe6
首先来重温一下printf格式控制的一些知识:
%x:无符号十六进制整数(不输出前缀0)
%2x:右对齐,当不足2位时,最左边用空格补齐。超过2位时,按实际长度显示
%02x:右对齐,当不足2位时,最左边用0补齐。超过2位时,按实际长度显示
从实际输出结果来看,实际长度为8位,超过了2位,按8位显示。
char其实是有符号的,其实发生了数据类型的转换,转换过程为,char(补码:e6)->int(补码:ffffffe6)->unsigned int(补码:ffffffe6),所以发生了实际的输出结果。
对程序略作修改,如下:
#include <stdio.h>
int main(int argc, char *argv[])
{
char a = 0xe6;
printf("a:%02x\n", (unsigned char)a);
return 0;
}
这时的输出就是:
a:e6