C语言谜题记录

 看完了C语言谜题,收获颇多,进一步理解了C语言,从其中列出的每一个例子中都可以学到不少以前被忽视的知识点。web

这里记录几个不错的case.安全

下面的程序会输出什么?ide

#include <stdio.h>
int main()
{
     float a = 12.5;
     printf ( "%d\n" , a);
     printf ( "%d\n" , ( int )a);
     printf ( "%d\n" , *( int *)&a);
     return 0;
}

参考答案
该项程序输出以下所示,
0
12
1095237632
缘由是:浮点数是4个字节,12.5f 转成二进制是:01000001010010000000000000000000,十六进制是:0×41480000,十进制是:1095237632。因此,第二和第三个输出相信你们也知道是为何了。而对于第一个,为何会输出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版:0×41480000 (在内存中是:00 00 48 41)
  • double版:0×4029000000000000 (在内存中是:00 00 00 00 00 00 29 40)

而咱们的%d要求是一个4字节的int,对于double的内存布局,咱们能够看到前四个字节是00,因此输出天然是0了。spa

这个示例向咱们说明printf并非类型安全的,这就是为何C++要引如cout的缘由了。code

 

ATTENTION:orm

    1,"hello"[2] == 2["hello"] = 'l'ip

    2,C/C++中,以0开头的数字都是八进制的。内存

    3,sizeof不是一个函数,是一个操做符,其求i++的类型的size,这是一件能够在程序运行前(编译时)彻底的事情,因此,sizeof(i++)直接就被4给取代了,在运行时也就不会有了i++这个表达式。

    4,switch-case体中的变量初始化语句不会被执行。

    5,printf返回值是输出的字符个数。

    6,stdout和stderr是否是同设备描述符。stdout是块设备,stderr则不是。对于块设备,只有当下面几种状况下才会被输入,1)遇到回车,2)缓冲区满,3)flush被调用。而stderr则不会。