1会输出"hello-std-out"?
#includestdio.h#includewindows.hintmain(){while(1){fprintf(stdout,"hello-std-out");fprintf(stderr,"hello-std-err");Sleep(1);}return0;}
参考答案:
stdout和stderr是不同设备描述符。stdout是块设备,stderr则不是。对于块设备,只有当下面几种情况下才会被输入:遇到回车;缓冲区满;flush被调用。而stderr则不会。
批注:
向stdout中输入会先将数据存放在输出缓冲区中,所以在上面的程序在缓冲区未满之前不会输出"hello-std-out"。
VS并没有测出这个效果,可能VS编译器做了优化或者某个选项没有打开。LinuxCentOS中可以看到。
2这段程序是有问题吗?
#includestdio.hintmain(){inta=1,2;printf("a:%d\n",a);return0;}
参考答案:
这个程序会得到编译出错(语法出错)。逗号表达式是没错,可是在初始化和变量声明时,逗号并不是逗号表达式的意义。这点要区分,要修改上面这个程序,你需要加上括号:"inta=(1,2);"。
3下面的程序会有什么样的输出呢?
#includestdio.hintmain(){inti=43;printf("%d\n",printf("%d",printf("%d",i)));return0;}
参考答案
程序会输出,你知道为什么吗?要知道为什么,你需要知道printf的返回值是什么。printf返回值是输出的字符个数。
4下面的程序会输出什么?
#includestdio.hintmain(){floata=12.5;printf("%d\n",a);printf("%d\n",(int)a);printf("%d\n",*(int*)a);return0;}
参考答案
该项程序输出:"2"。
原因是:浮点数是4个字节,12.5f转成二进制是:,十六进制是:0x,十进制是:。所以,第二和第三个输出相信大家也知道是为什么了。
而对于第一个,为什么会输出0,我们需要了解一下float和double的内存布局,如下:
float:1位符号位(s)、8位指数(e),23位尾数(m,共32位)。
double:1位符号位(s)、11位指数(e),52位尾数(m,共64位)。
然后,我们还需要了解一下printf由于类型不匹配,所以,会把float直接转成double,注意,12.5的float和double的内存二进制完全不一样。别忘了在x86芯片下使用是的反字节序,高位字节和低位字位要反过来。所以:
float版:0x(在内存中是:)。
double版:0x4029(在内存中是:)。
而我们的%d要求是一个4字节的int,对于double的内存布局,我们可以看到前四个字节是00,所以输出自然是0了。这个示例向我们说明printf并不是类型安全的,这就是为什么C++要引如cout的原因了。
批注:
参考文章: