你真的精通C語言嗎?來解這十道C語言迷題試試吧!

本文展現了10個C語言的迷題以及答案,並且有至關的一些例子多是咱們平常工做可能會見獲得的。經過這些迷題,但願你能更瞭解C語言。程序員

若是你不看答案,不知道是否有把握回答各個謎題?讓咱們來試試。編程


 

一、會輸出"hello-std-out"?

#include<stdio.h>數組

int main()安全

{編程語言

while (1)函數

    {佈局

        fprintf(stdout, "hello-std-out");學習

        fprintf(stderr, "hello-std-err");this

        sleep(1);spa

    }

    return 0;

}

參考答案

stdout和stderr是不一樣設備描述符。stdout是塊設備,stderr則不是。對於塊設備,只有當下面幾種狀況下才會被輸入:遇到回車;緩衝區滿;flush被調用。而stderr則不會。

二、這段程序是有問題嗎?

#include<stdio.h>

int main()

{

    int a = 1,2;

    printf("a : %d\n",a);

    return 0;

}

參考答案

這個程序會獲得編譯出錯(語法出錯)。逗號表達式是沒錯,但是在初始化和變量聲明時,逗號並非逗號表達式的意義。這點要區分,要修改上面這個程序,你須要加上括號:"int a = (1,2);"。

三、下面的程序會有什麼樣的輸出呢?

#include<stdio.h>

int main()

{

    int i=43;

    printf("%d\n",printf("%d",printf("%d",i)));

    return 0;

}

參考答案

程序會輸出4321,你知道爲何嗎?要知道爲何,你須要知道printf的返回值是什麼。printf返回值是輸出的字符個數。

四、下面的程序會輸出什麼?

#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,十六進制是:0x41480000,十進制是: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版:0x41480000 (在內存中是:00 00 48 41)。

double版:0x4029000000000000 (在內存中是:00 00 00 00 00 00 29 40)。

而咱們的%d要求是一個4字節的int,對於double的內存佈局,咱們能夠看到前四個字節是00,因此輸出天然是0了。這個示例向咱們說明printf並非類型安全的,這就是爲何C++要引如cout的緣由了。


若是你學習C/C++遇到瓶頸,迷茫,困惑,那麼不妨加入小編的學習企鵝圈子,跟着前輩一塊兒交流學習,永遠會比單打獨鬥強得多!


五、下面的程序輸出是多少?並解釋爲何?

#include<stdio.h>

int main()

{

int a = 1;

switch (a)

{

int b = 20;

case 1:

printf("b is %d\n", b);

break;

default:

printf("b is %d\n", b);

break;

}

return 0;

}

參考答案

該程序在編譯時,報錯:「b」的初始化操做由「case」標籤跳過,「default」標籤跳過「b」的初始化操做

六、下面的程序會有什麼潛在的危險?

#include<stdio.h>

int main()

{

char str[10];

printf("Enter the string:");

scanf("%s", str);

printf("You entered:%s\n", str);

return 0;

}

參考答案

本題很簡單了。這個程序的潛在問題是,若是用戶輸入了超過80個長度的字符,那麼就會有數組越界的問題了,你的程序頗有可能會crash了。

七、請問下面的程序輸出什麼?

#include<stdio.h>

int main()

{

int i;

i = 10;

printf("i : %d\n", i);

printf("sizeof(i++) is: %d\n", sizeof(i++));

printf("i : %d\n", i);

return 0;

}

參考答案

若是你以爲輸出分別是:10,4,11。那麼你就錯了。

錯在了第三個,第一個是10沒有什麼問題,第二個是4,也沒有什麼問題,由於是32位機上一個int有4個字節。可是第三個爲何輸出的不是11呢?竟然仍是10?緣由是,sizeof不是一個函數,是一個操做符,其求i++的類型的size,這是一件能夠在程序運行前(編譯時)徹底的事情,因此,sizeof(i++)直接就被4給取代了,在運行時也就不會有了i++這個表達式。

八、下面的程序的輸出值是什麼?

#include<stdio.h>

#define SIZEOF(arr) (sizeof(arr)/sizeof(arr[0]))

#define PrintInt(expr) printf("%s:%d\n",#expr,(expr))

int main()

{

/* The powers of 10 */

int pot[] = {

0001,

0010,

0100,

1000

};

int i;

for (i = 0; i < SIZEOF(pot); i++)PrintInt(pot[i]);

return 0;

}

參考答案

若是你對於PrintInt這個宏有問題的話,能夠去看一看資料。不過,本例的問題不在這裏,本例的輸出會是:1,8,64,1000。其實很簡單了,在C/C++中,以0開頭的數字都是八進制的。

九、下面的輸出是什麼?

#include<stdio.h>

int main()

{

int i = 6;

if (((++i < 7) && (i++ / 6)) || (++i <= 9));

printf("%d\n", i);

return 0;

}

參考答案

本題並不簡單的是考前綴++或反綴++,本題主要考的是&&和||的短路求值的問題。

所謂短路求值:對於(條件1 && 條件2),若是「條件1」是false,那「條件2」的表達式會被忽略了。對於(條件1 || 條件2),若是「條件1」爲true,而「條件2」的表達式則被忽略了。

因此,我相信你會知道本題的答案是什麼了。

十、下面的C程序是合法的嗎?

#include<stdio.h>

int main()

{

int a = 3, b = 5;

printf(&a["Ya!Hello! how is this? %s\n"], &b["junk/super"]);

//等價printf("Hello! how is this? %s\n", "super");

printf(&a["WHAT%c%c%c %c%c %c !\n"], 1["this"], 2["beauty"], 0["tool"], 0["is"], 3["sensitive"], 4["CCCCCC"]);

return 0;

}

參考答案

本例是合法的,輸出爲:"Hello! how is this? super That is C !"

本例主要展現了一種另類的用法。下面的兩種用法是相同的:

"hello"[2]

2["hello"]

若是你知道:a[i] 其實就是 *(a+i)也就是 *(i+a),因此若是寫成 i[a] 應該也不難理解了。


 

最後,若是你也想成爲程序員,想要快速掌握編程,趕忙加入學習企鵝圈子!

裏面有資深專業軟件開發工程師,在線解答你的全部疑惑~編程語言入門「so easy」

編程學習書籍:


 

編程學習視頻:

相關文章
相關標籤/搜索