沒(méi)想到吧,一行注釋也能影響運(yùn)行結(jié)果
也許你在某個(gè)段子里聽(tīng)說(shuō)過(guò),某行注釋刪掉后,程序竟然不能預(yù)期執(zhí)行?真的會(huì)這樣么?你還別不信。
見(jiàn)證“奇跡”
1#include <stdio.h>
2#include <math.h>
3int main(void)
4{
5 int a = (int)sqrt(30);
6 //is sqrt(30) + 1 > 5 ??/
7 //故意讓a++,你別管為什么
8 a++;
9 if(a > 5)
10 printf("sqrt(30) +1 > 5 \n");
11 else
12 {
13 printf("sqrt(30) +1 <= 5 \n");
14 }
15 return 0;
16}
編譯運(yùn)行:
1$ gcc -o main main.c -trigraphs
2$ ./main
3sqrt(30) +1 > 5
作為受過(guò)九年義務(wù)教育的我們,一看就知道結(jié)果是對(duì)的,畢竟 5 * 5 < 30 < 6 * 6。
但是刪掉第九行,再運(yùn)行:
1$ ./main
2sqrt(30) +1 <= 5
結(jié)果竟然變了??!
看到這里,細(xì)心的讀者可能已經(jīng)發(fā)現(xiàn)了其中的端倪,我也就不賣(mài)關(guān)子了。接下來(lái)就說(shuō)說(shuō)我們本次要提到的主角-三字符組(trigraph sequences)。
三字符組我們都知道C語(yǔ)言里面有很多轉(zhuǎn)義符號(hào),例如:
1\n 換行符(LF)
2\r 回車(chē)符(CR)
3\t 水平制表符(HT)
4\b 退格符(BS)
5\’ 單引號(hào)
6\” 雙引號(hào)
7\\ 反斜杠
8......
當(dāng)然還有很多,我就不一一列舉了。這些符號(hào)在代碼中都有特別的作用,或者無(wú)法直接輸入,因此用轉(zhuǎn)移符+其他字符組合來(lái)代替。
同樣的,早期的一些鍵盤(pán)可能沒(méi)法輸入一些特殊的符號(hào),如:
1# $ @ [ \ ] ^ ` { | } ~
于是,為了解決這個(gè)問(wèn)題C語(yǔ)言標(biāo)準(zhǔn)規(guī)定預(yù)處理器(C preprocessor)在掃描處理C語(yǔ)言源文件時(shí),替換下述的3字符出現(xiàn)為1個(gè)字符:
也就是說(shuō),??/會(huì)被替換為\,如果刪掉原先的第九行,就變成了:
1 //is sqrt(30) + 1 > 5 \
2 a++;
我們都知道,\的存在(通常一行代碼太長(zhǎng),可以通過(guò)該符號(hào)來(lái)?yè)Q行),使得上面看似兩行,實(shí)則是一行。即變成了:
1 //is sqrt(30) + 1 > 5 a++;
也就是說(shuō),a++根本不會(huì)執(zhí)行了,當(dāng)然會(huì)導(dǎo)致最終結(jié)果不符合預(yù)期。
當(dāng)然了,很多現(xiàn)代編譯器可能并不會(huì)做這樣的替換,所以這樣的問(wèn)題也基本無(wú)需擔(dān)心,老實(shí)用原本的符號(hào)即可。 實(shí)際上,細(xì)心的讀者可能觀察到了,我在前面例子代碼中加了編譯選項(xiàng)-trigraphs,否則的話,編譯是有警告的:
1$ gcc -o main main.c
2main.c: In function ‘main’:
3main.c:6:27: warning: trigraph ??/ ignored, use -trigraphs to enable [-Wtrigraphs]
4 //is sqrt(30) + 1 > 5 ??/
雙字符組除了三字符組,還有雙字符組。
總結(jié)今天的你不知道很難踩坑,知道了也沒(méi)啥用的內(nèi)容就介紹到這里了。
獲取更多:點(diǎn)擊文章下面的鏈接~