欧美性猛交XXXX免费看蜜桃,成人网18免费韩国,亚洲国产成人精品区综合,欧美日韩一区二区三区高清不卡,亚洲综合一区二区精品久久

打開(kāi)APP
userphoto
未登錄

開(kāi)通VIP,暢享免費電子書(shū)等14項超值服

開(kāi)通VIP
C語(yǔ)言的謎題 | 酷殼 - CoolShell.cn


這幾天,本站推出了幾篇關(guān)于C語(yǔ)言的很多文章如下所示:

我們可以看到很多C語(yǔ)言相關(guān)的一些東西。比如《語(yǔ)言的歧義》主要告訴了大家C語(yǔ)言中你意想不到的錯誤以及一些歧義上的東西。而《誰(shuí)說(shuō)C語(yǔ)言很簡(jiǎn)單》則通過(guò)一些看似你從來(lái)不可能寫(xiě)出的代碼來(lái)告訴大家C語(yǔ)言并不是一件容易事情?!?個(gè)變態(tài)的hello world》和《如何弄亂C的源代碼》則以一種極端的方式告訴大家,不要以為咱們自己寫(xiě)不出混亂的代碼,每個(gè)程序員其實(shí)都有把代碼搞得一團亂的潛質(zhì)。通過(guò)這些文章,相信你對編程或是你覺(jué)得很簡(jiǎn)單的C語(yǔ)言有了一些了解。是的,很不容易吧,以前是不是低估了編程和C語(yǔ)言?今天是否我們又在低估C++和Java呢?

本篇文章《C語(yǔ)言的謎題》展示了14個(gè)C語(yǔ)言的迷題以及答案,代碼應該是足夠清楚的,而且我也相信有相當的一些例子可能是我們日常工作可能會(huì )見(jiàn)得到的。通過(guò)這些迷題,希望你能更了解C語(yǔ)言。如果你不看答案,不知道是否有把握回答各個(gè)謎題?讓我們來(lái)試試。

1、下面的程序并不見(jiàn)得會(huì )輸出 hello-std-out,你知道為什么嗎?

01 #include <stdio.h>
02 #include <unistd.h>
03 int main()
04 {
05     while(1)
06     {
07         fprintf(stdout,"hello-std-out");
08         fprintf(stderr,"hello-std-err");
09         sleep(1);
10     }
11     return 0;
12 }

參考答案:stdout和stderr是不是同設備描述符。stdout是塊設備,stderr則不是。對于塊設備,只有當下面幾種情況下才會(huì )被輸入,1)遇到回車(chē),2)緩沖區滿(mǎn),3)flush被調用。而stderr則不會(huì )。

2、下面的程序看起來(lái)是正常的,使用了一個(gè)逗號表達式來(lái)做初始化。可惜這段程序是有問(wèn)題的。你知道為什么呢?

1 #include <stdio.h>
2   
3 int main()
4 {
5     int a = 1,2;
6     printf("a : %d\n",a);
7     return 0;
8 }

參考答案:這個(gè)程序會(huì )得到編譯出錯(語(yǔ)法出錯),逗號表達式是沒(méi)錯,可是在初始化和變量聲明時(shí),逗號并不是逗號表達式的意義。這點(diǎn)要區分,要修改上面這個(gè)程序,你需要加上括號: int a = (1,2);

3、下面的程序會(huì )有什么樣的輸出呢?

1 #include <stdio.h>
2 int main()
3 {
4     int i=43;
5     printf("%d\n",printf("%d",printf("%d",i)));
6     return 0;
7 }

參考答案:程序會(huì )輸出4321,你知道為什么嗎?要知道為什么,你需要知道printf的返回值是什么。printf返回值是輸出的字符個(gè)數。

4、下面的程序會(huì )輸出什么?

1 #include <stdio.h>
2 int main()
3 {
4     float a = 12.5;
5     printf("%d\n", a);
6     printf("%d\n", (int)a);
7     printf("%d\n", *(int *)&a);
8     return 0;
9 }

參考答案
該項程序輸出如下所示,
0
12
1095237632
原因是:浮點(diǎn)數是4個(gè)字節,12.5f 轉成二進(jìn)制是:01000001010010000000000000000000,十六進(jìn)制是:0×41480000,十進(jìn)制是:1095237632。所以,第二和第三個(gè)輸出相信大家也知道是為什么了。而對于第一個(gè),為什么會(huì )輸出0,我們需要了解一下float和double的內存布局,如下:

  • float: 1位符號位(s)、8位指數(e),23位尾數(m,共32位)
  • double: 1位符號位(s)、11位指數(e),52位尾數(m,共64位)

然后,我們還需要了解一下printf由于類(lèi)型不匹配,所以,會(huì )把float直接轉成double,注意,12.5的float和double的內存二進(jìn)制完全不一樣。別忘了在x86芯片下使用是的反字節序,高位字節和低位字位要反過(guò)來(lái)。所以:

  • float版:0×41480000 (在內存中是:00 00 48 41)
  • double版:0×4029000000000000 (在內存中是:00 00 00 00 00 00 29 40)

而我們的%d要求是一個(gè)4字節的int,對于double的內存布局,我們可以看到前四個(gè)字節是00,所以輸出自然是0了。

這個(gè)示例向我們說(shuō)明printf并不是類(lèi)型安全的,這就是為什么C++要引如cout的原因了。

5、下面,我們再來(lái)看一個(gè)交叉編譯的事情,下面的兩個(gè)文件可以編譯通過(guò)嗎?如果可以通過(guò),結果是什么?

file1.c

file2.c

1 extern int *arr;
2 int main()
3 {
4     arr[1] = 100;
5     printf("%d\n", arr[1]);
6     return 0;
7 }

參考答案:該程序可以編譯通過(guò),但運行時(shí)會(huì )出錯。為什么呢?原因是,在另一個(gè)文件中用 extern int *arr來(lái)外部聲明一個(gè)數組并不能得到實(shí)際的期望值,因為他們的類(lèi)型并不匹配。所以導致指針實(shí)際并沒(méi)有指向那個(gè)數組。注意:一個(gè)指向數組的指針,并不等于一個(gè)數組。修改:extern int arr[]。(參考:ISO C語(yǔ)言 6.5.4.2 節)

6、請說(shuō)出下面的程序輸出是多少?并解釋為什么?(注意,該程序并不會(huì )輸出 “b is 20″)

01 #include <stdio.h>
02 int main()
03 {
04     int a=1;
05     switch(a)
06     {
07         int b=20;
08         case 1:
09             printf("b is %d\n",b);
10             break;
11         default:
12             printf("b is %d\n",b);
13             break;
14     }
15     return 0;
16 }

參考答案:該程序在編譯時(shí),可能會(huì )出現一條warning: unreachable code at beginning of switch statement。我們以為進(jìn)入switch后,變量b會(huì )被初始化,其實(shí)并不然,因為switch-case語(yǔ)句會(huì )把變量b的初始化直接就跳過(guò)了。所以,程序會(huì )輸出一個(gè)隨機的內存值。

7、請問(wèn)下面的程序會(huì )有什么潛在的危險?

1 #include <stdio.h>
2 int main()
3 {
4     char str[80];
5     printf("Enter the string:");
6     scanf("%s",str);
7     printf("You entered:%s\n",str);
8     return 0;
9 }

參考答案:本題很簡(jiǎn)單了。這個(gè)程序的潛在問(wèn)題是,如果用戶(hù)輸入了超過(guò)80個(gè)長(cháng)度的字符,那么就會(huì )有數組越界的問(wèn)題了,你的程序很有可以及會(huì )crash了。

8、請問(wèn)下面的程序輸出什么?

01 #include <stdio.h>
02 int main()
03 {
04     int i;
05     i = 10;
06     printf("i : %d\n",i);
07     printf("sizeof(i++) is: %d\n",sizeof(i++));
08     printf("i : %d\n",i);
09     return 0;
10 }

參考答案:如果你覺(jué)得輸出分別是,10,4,11,那么你就錯了,錯在了第三個(gè),第一個(gè)是10沒(méi)有什么問(wèn)題,第二個(gè)是4,也沒(méi)有什么問(wèn)題,因為是32位機上一個(gè)int有4個(gè)字節。但是第三個(gè)為什么輸出的不是11呢?居然還是10?原因是,sizeof不是一個(gè)函數,是一個(gè)操作符,其求i++的類(lèi)型的size,這是一件可以在程序運行前(編譯時(shí))完全的事情,所以,sizeof(i++)直接就被4給取代了,在運行時(shí)也就不會(huì )有了i++這個(gè)表達式。

9、請問(wèn)下面的程序的輸出值是什么?

01 #include <stdio.h>
02 #include <stdlib.h>
03   
04 #define SIZEOF(arr) (sizeof(arr)/sizeof(arr[0]))
05 #define PrintInt(expr) printf("%s:%d\n",#expr,(expr))
06   
07 int main()
08 {
09     /* The powers of 10 */
10     int pot[] = {
11                     0001,
12                     0010,
13                     0100,
14                     1000
15                 };
16   
17     int i;
18     for(i=0;i<SIZEOF(pot);i++)
19         PrintInt(pot[i]);
20   
21     return 0;
22 }

參考答案:好吧,如果你對于PrintInt這個(gè)宏有問(wèn)題的話(huà),你可以去看一看《語(yǔ)言的歧義》中的第四個(gè)示例。不過(guò),本例的問(wèn)題不在這里,本例的輸出會(huì )是:1,8,64,1000,其實(shí)很簡(jiǎn)單了,以C/C++中,以0開(kāi)頭的數字都是八進(jìn)制的。

10、請問(wèn)下面的程序輸出是什么?(絕對不是10)

#include #define PrintInt(expr) printf("%s : %dn",#expr,(expr))int main(){int y = 100;int *p;p = malloc(sizeof(int));*p = 10;y = y/*p; /*dividing y by *p */;PrintInt(y);return 0;}

參考答案:本題輸出的是100。為什么呢?問(wèn)題就出在 y = y/*p;上了,我們本來(lái)想的是 y / (*p) ,然而,我們沒(méi)有加入空格和括號,結果y/*p中的 /*被解釋成了注釋的開(kāi)始。于是,這也是整個(gè)惡夢(mèng)的開(kāi)始。

11、下面的輸出是什么?

01 #include <stdio.h>
02 int main()
03 {
04     int i = 6;
05     if( ((++i < 7) && ( i++/6)) || (++i <= 9))
06         ;
07   
08     printf("%d\n",i);
09     return 0;
10 }

參考答案:本題并不簡(jiǎn)單的是考前綴++或反綴++,本題主要考的是&&和||的短路求值的問(wèn)題。所為短路求值:對于(條件1 && 條件2),如果“條件1”是false,那“條件2”的表達式會(huì )被忽略了。對于(條件1 || 條件2),如果“條件1”為true,而“條件2”的表達式則被忽略了。所以,我相信你會(huì )知道本題的答案是什么了。

12、下面的C程序是合法的嗎?如果是,那么輸出是什么?

01 #include <stdio.h>
02 int main()
03 {
04     int a=3, b = 5;
05   
06     printf(&a["Ya!Hello! how is this? %s\n"], &b["junk/super"]);
07   
08     printf(&a["WHAT%c%c%c  %c%c  %c !\n"], 1["this"],
09         2["beauty"],0["tool"],0["is"],3["sensitive"],4["CCCCCC"]);
10   
11     return 0;
12 }

參考答案
本例是合法的,輸出如下:

Hello! how is this? super
That is C !

本例主要展示了一種另類(lèi)的用法。下面的兩種用法是相同的:

“hello”[2]
2["hello"]

如果你知道:a[i] 其實(shí)就是 *(a+i)也就是 *(i+a),所以如果寫(xiě)成 i[a] 應該也不難理解了。

13、請問(wèn)下面的程序輸出什么?(假設:輸入 Hello, World)

1 #include <stdio.h>
2 int main()
3 {
4     char dummy[80];
5     printf("Enter a string:\n");
6     scanf("%[^r]",dummy);
7     printf("%s\n",dummy);
8     return 0;
9 }

參考答案:本例的輸出是“Hello, Wo”,scanf中的”%[^r]“是從中作梗的東西。意思是遇到字符r就結束了。

14、下面的程序試圖使用“位操作”來(lái)完成“乘5”的操作,不過(guò)這個(gè)程序中有個(gè)BUG,你知道是什么嗎?

01 #include <stdio.h>
02 #define PrintInt(expr) printf("%s : %d\n",#expr,(expr))
03 int FiveTimes(int a)
04 {
05     int t;
06     t = a<<2 + a;
07     return t;
08 }
09   
10 int main()
11 {
12     int a = 1, b = 2,c = 3;
13     PrintInt(FiveTimes(a));
14     PrintInt(FiveTimes(b));
15     PrintInt(FiveTimes(c));
16     return 0;
17 }

參考答案:本題的問(wèn)題在于函數FiveTimes中的表達式“t = a<<2 + a;”,對于a<<2這個(gè)位操作,優(yōu)先級要比加法要低,所以這個(gè)表達式就成了“t = a << (2+a)”,于是我們就得不到我們想要的值。該程序修正如下:

1 int FiveTimes(int a)
2 {
3     int t;
4     t = (a<<2) + a;
5     return t;
6 }
本站僅提供存儲服務(wù),所有內容均由用戶(hù)發(fā)布,如發(fā)現有害或侵權內容,請點(diǎn)擊舉報。
打開(kāi)APP,閱讀全文并永久保存 查看更多類(lèi)似文章
猜你喜歡
類(lèi)似文章
高級語(yǔ)言程序設計A第4次作業(yè)
兩個(gè)超級有意思的C語(yǔ)言題目,很多人都一臉懵逼
C語(yǔ)言的數組學(xué)習入門(mén)之對數組初始化的操作
C/C++編程知識:運算符(五)丨sizeof()運算符來(lái)啦!
初學(xué)C語(yǔ)言-循環(huán)
C語(yǔ)言中數組的總結
更多類(lèi)似文章 >>
生活服務(wù)
分享 收藏 導長(cháng)圖 關(guān)注 下載文章
綁定賬號成功
后續可登錄賬號暢享VIP特權!
如果VIP功能使用有故障,
可點(diǎn)擊這里聯(lián)系客服!

聯(lián)系客服

欧美性猛交XXXX免费看蜜桃,成人网18免费韩国,亚洲国产成人精品区综合,欧美日韩一区二区三区高清不卡,亚洲综合一区二区精品久久