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則不會。

相關文章
相關標籤/搜索