當(dāng)前位置:首頁(yè) > 公眾號(hào)精選 > 程序員寫(xiě)個(gè)解
[導(dǎo)讀]從匯編角度來(lái)說(shuō),如果“test %al, %al”能改成“test %0x1, %al”就沒(méi)有匪夷所思的問(wèn)題了,如此一來(lái)應(yīng)該會(huì)降低CPU的效率,畢竟執(zhí)行指令還需要一個(gè)立即數(shù),我沒(méi)搞過(guò)編譯器也沒(méi)設(shè)計(jì)過(guò)CPU,純屬瞎猜,能搞編譯器的家伙都是大牛的存在,咱們吃瓜的參合個(gè)啥!

上一篇文章《C語(yǔ)言bool占用4個(gè)字節(jié)?匯編之下無(wú)秘密|帶你看extern》分析在C99標(biāo)準(zhǔn)下bool類型占用1Byte,而不是1bit,C語(yǔ)言 不存在內(nèi)存長(zhǎng)度小于8bit的數(shù)據(jù)類型,思考:

1、如果bool類型高7bit不是0,使用bool類型是否出現(xiàn)匪夷所思的結(jié)果?

2、執(zhí)行if判斷bool類型時(shí),它判斷的是所有8比特?還是最低比特?

接下來(lái)我分享一個(gè)奇特的案例現(xiàn)象,并在從反匯編角度去解釋現(xiàn)象產(chǎn)生原因。

1. 俄羅斯轉(zhuǎn)輪

玩?zhèn)€勇敢者的賭槍游戲——俄羅斯轉(zhuǎn)輪。

左輪手槍彈槽篩入一顆子彈,快速旋轉(zhuǎn)彈槽,合上彈槽,朝著對(duì)方腦袋開(kāi)槍,活下來(lái)的勝利。接下來(lái)友請(qǐng)賭槍游戲必勝客“燕雙鷹”。

燕雙鷹:“我有個(gè)習(xí)慣,會(huì)殺死向自己開(kāi)槍的人,哪怕他的槍里沒(méi)有子彈……”

我:“等等燕大俠,沒(méi)搶、沒(méi)搶。解放了70年咯,1966年在大會(huì)堂玻璃被子彈擊穿事件后,周總理就下達(dá)指令全民禁槍、民眾自愿上繳槍械?!?/span>

燕雙鷹:“那為什么請(qǐng)我出場(chǎng)?”

我:“21世紀(jì)國(guó)家科研、資本家壓榨都講成本,沒(méi)有槍械,可以模擬呀。自動(dòng)駕駛不一定都需要先造車再去馬路上跑,完全能建立3D場(chǎng)景,在游戲虛擬環(huán)境下訓(xùn)練自動(dòng)駕駛算法。同樣賭槍游戲也能模擬。

“子彈放在8bit寄存器里,寄存器相當(dāng)于彈槽,最低比特相當(dāng)于蓄勢(shì)待發(fā)的子彈。下面是游戲的源代碼?!?/span>


傳入0:表示搶里沒(méi)有子彈。

傳入1:表示子彈在第1激發(fā)位置。

傳入2:表示子彈在第2激發(fā)位置。

傳入4:表示子彈在第3激發(fā)位置。


燕雙鷹:“明白,來(lái)~咱們弄點(diǎn)刺激的,隨機(jī)放入2顆子彈如何,編劇從來(lái)沒(méi)允許賭搶上輸過(guò)?!?/span>

我:“大俠且慢,暖男郭先生說(shuō)沖動(dòng)是魔鬼,咱們1顆子彈試試水?!?/span>

篩入1顆子彈,子彈落入第2激發(fā)位置,扣動(dòng)扳機(jī),屏幕上顯示“false:燕雙鷹贏”。燕雙鷹臉上漏出招牌式微笑。

下一刻屏幕緊跟著輸出“true:Bang 燕雙鷹你輸了”,燕雙鷹眉頭顯出深深的“川”字紋。

各位看官,你能想到燕雙鷹中彈原因嗎?當(dāng)然,如果你能保證絕對(duì)不會(huì)往布爾類型傳遞0/1以外的值,本文不用繼續(xù)往下讀。

all: @gcc bool-char.c -g @objdump a.out -S > a.dis @./a.out 0 @./a.out 1 @./a.out 2 @./a.out 3

2. 匯編解釋

接下來(lái)解釋燕雙鷹為什么會(huì)輸。

同樣的代碼在x86、ARM、mips架構(gòu)下用gcc編譯,執(zhí)行結(jié)果都一樣,至于匯編我只解釋x86架構(gòu)下的指令。

兩條件表達(dá)式的匯編都差不多,唯一區(qū)別是第一條多一個(gè)異或指令。


movzbl -0x9(%rbp),%eax:以4Byte方式載入數(shù)據(jù)到eax寄存器,eax是32bit寄存器,eax存儲(chǔ)的是彈槽子彈位置。

test %al, %al:al寄存器的值和它自己“與”操作,al是eax的低8bit寄存器。只要al寄存器8bit不全為0,則返回真。

test指令和and指令都是執(zhí)行“與”操作,不過(guò)test指令會(huì)影響3個(gè)標(biāo)志位:SF(執(zhí)行后數(shù)據(jù)的正負(fù))、ZF(執(zhí)行后結(jié)果是否為0)、PF(執(zhí)行后二進(jìn)制1的個(gè)數(shù)是否為偶數(shù)),and指令不會(huì)修改他們, 本文關(guān)注的是ZF標(biāo)志位。

xor $0x1,%eax:僅對(duì)eax寄存器的最低比特執(zhí)行異或。

C代碼“if(!a)”的感嘆號(hào)“!”被編譯器翻譯成xor和test的組合。注意到了嗎,只要eax不是0或1,兩條指令都會(huì)執(zhí)行。

2.1. 執(zhí)行if(!a)

如果eax=0x00,則xor結(jié)果eax=0x01;test返回真

如果eax=0x01,則xor結(jié)果eax=0x00;test返回假

如果eax=0x02,則xor結(jié)果eax=0x03;test返回

2.2. 執(zhí)行if(a)

如果eax=0x00,test返回假

如果eax=0x01,test返回真

如果eax=0x02,test返回

3. 小白才寫(xiě)得出的代碼

看官或許會(huì)想:“正常情況誰(shuí)會(huì)這么寫(xiě)例子上的垃圾代碼,往bool傳遞0/1以外的數(shù)據(jù),八成是作者為了水文章瞎弄文案?!?/span>

“No No No。”

6年前我曾今寫(xiě)過(guò)一個(gè)C函數(shù),函數(shù)需要傳遞bool類型“指針”。在同事眼里:“布爾類型嘛,懂~,老熟人咯?!?/span>

于是,他強(qiáng)制轉(zhuǎn)換char為bool,向我的函數(shù)傳遞變量指針。

絕大多數(shù)C語(yǔ)言學(xué)習(xí)者的實(shí)操平臺(tái)要么是Keil C51、要么是Trubo C,兩個(gè)編譯環(huán)境都使用C89標(biāo)準(zhǔn),按照C89的套路,bool類型通常都是重新定義char得來(lái)(typedef char bool),殊不知bool類型已經(jīng)被C99正式收編,GCC也給它名份,成了C語(yǔ)言家族的第9房小妾(其他妻妾包括char、short、int、long、float、double、void、指針)。


								
void fun(bool *a){ if (!*a) { printf("false\r\n"); } if (*a) { printf("true\r\n"); }}int main(int argc, char **argv) { char in = 2;  fun((bool*)&in); return 0;}

若同事規(guī)規(guī)矩矩的向布爾類型賦值0(false)或1(true)還好,可誰(shuí)曾想到他某次傳遞一個(gè)2進(jìn)去,一個(gè)表達(dá)式憑什么既可能是true、也同時(shí)是false呢?

$ ./a.out falsetrue

猜測(cè)同事把布爾類型和布爾表達(dá)式搞混了:

布爾類型:只觀察最低比特

布爾表達(dá)式:非0即是真。


4. 指令修改

從匯編角度來(lái)說(shuō),如果“test %al, %al”能改成“test %0x1, %al”就沒(méi)有匪夷所思的問(wèn)題了,如此一來(lái)應(yīng)該會(huì)降低CPU的效率,畢竟執(zhí)行指令還需要一個(gè)立即數(shù),我沒(méi)搞過(guò)編譯器也沒(méi)設(shè)計(jì)過(guò)CPU,純屬瞎猜,能搞編譯器的家伙都是大牛的存在,咱們吃瓜的參合個(gè)啥!

本站聲明: 本文章由作者或相關(guān)機(jī)構(gòu)授權(quán)發(fā)布,目的在于傳遞更多信息,并不代表本站贊同其觀點(diǎn),本站亦不保證或承諾內(nèi)容真實(shí)性等。需要轉(zhuǎn)載請(qǐng)聯(lián)系該專欄作者,如若文章內(nèi)容侵犯您的權(quán)益,請(qǐng)及時(shí)聯(lián)系本站刪除。
換一批
延伸閱讀

9月2日消息,不造車的華為或?qū)⒋呱龈蟮莫?dú)角獸公司,隨著阿維塔和賽力斯的入局,華為引望愈發(fā)顯得引人矚目。

關(guān)鍵字: 阿維塔 塞力斯 華為

倫敦2024年8月29日 /美通社/ -- 英國(guó)汽車技術(shù)公司SODA.Auto推出其旗艦產(chǎn)品SODA V,這是全球首款涵蓋汽車工程師從創(chuàng)意到認(rèn)證的所有需求的工具,可用于創(chuàng)建軟件定義汽車。 SODA V工具的開(kāi)發(fā)耗時(shí)1.5...

關(guān)鍵字: 汽車 人工智能 智能驅(qū)動(dòng) BSP

北京2024年8月28日 /美通社/ -- 越來(lái)越多用戶希望企業(yè)業(yè)務(wù)能7×24不間斷運(yùn)行,同時(shí)企業(yè)卻面臨越來(lái)越多業(yè)務(wù)中斷的風(fēng)險(xiǎn),如企業(yè)系統(tǒng)復(fù)雜性的增加,頻繁的功能更新和發(fā)布等。如何確保業(yè)務(wù)連續(xù)性,提升韌性,成...

關(guān)鍵字: 亞馬遜 解密 控制平面 BSP

8月30日消息,據(jù)媒體報(bào)道,騰訊和網(wǎng)易近期正在縮減他們對(duì)日本游戲市場(chǎng)的投資。

關(guān)鍵字: 騰訊 編碼器 CPU

8月28日消息,今天上午,2024中國(guó)國(guó)際大數(shù)據(jù)產(chǎn)業(yè)博覽會(huì)開(kāi)幕式在貴陽(yáng)舉行,華為董事、質(zhì)量流程IT總裁陶景文發(fā)表了演講。

關(guān)鍵字: 華為 12nm EDA 半導(dǎo)體

8月28日消息,在2024中國(guó)國(guó)際大數(shù)據(jù)產(chǎn)業(yè)博覽會(huì)上,華為常務(wù)董事、華為云CEO張平安發(fā)表演講稱,數(shù)字世界的話語(yǔ)權(quán)最終是由生態(tài)的繁榮決定的。

關(guān)鍵字: 華為 12nm 手機(jī) 衛(wèi)星通信

要點(diǎn): 有效應(yīng)對(duì)環(huán)境變化,經(jīng)營(yíng)業(yè)績(jī)穩(wěn)中有升 落實(shí)提質(zhì)增效舉措,毛利潤(rùn)率延續(xù)升勢(shì) 戰(zhàn)略布局成效顯著,戰(zhàn)新業(yè)務(wù)引領(lǐng)增長(zhǎng) 以科技創(chuàng)新為引領(lǐng),提升企業(yè)核心競(jìng)爭(zhēng)力 堅(jiān)持高質(zhì)量發(fā)展策略,塑強(qiáng)核心競(jìng)爭(zhēng)優(yōu)勢(shì)...

關(guān)鍵字: 通信 BSP 電信運(yùn)營(yíng)商 數(shù)字經(jīng)濟(jì)

北京2024年8月27日 /美通社/ -- 8月21日,由中央廣播電視總臺(tái)與中國(guó)電影電視技術(shù)學(xué)會(huì)聯(lián)合牽頭組建的NVI技術(shù)創(chuàng)新聯(lián)盟在BIRTV2024超高清全產(chǎn)業(yè)鏈發(fā)展研討會(huì)上宣布正式成立。 活動(dòng)現(xiàn)場(chǎng) NVI技術(shù)創(chuàng)新聯(lián)...

關(guān)鍵字: VI 傳輸協(xié)議 音頻 BSP

北京2024年8月27日 /美通社/ -- 在8月23日舉辦的2024年長(zhǎng)三角生態(tài)綠色一體化發(fā)展示范區(qū)聯(lián)合招商會(huì)上,軟通動(dòng)力信息技術(shù)(集團(tuán))股份有限公司(以下簡(jiǎn)稱"軟通動(dòng)力")與長(zhǎng)三角投資(上海)有限...

關(guān)鍵字: BSP 信息技術(shù)
關(guān)閉