C_and_CPP 板


LINE

原文: https://mohitmv.github.io/blog/Shocking-Undefined-Behaviour-In-Action/ 看到一篇 C++ 未定義行為的簡單例子分享一下 簡單的有限迴圈變成無窮迴圈 1. #include <iostream> 2. 3. int main() { 4. char buf[50] = "y"; 5. for (int j = 0; j < 9; ++j) { 6. std::cout << (j * 0x20000001) << std::endl; 7. if (buf[0] == 'x') break; 8. } 9. } 編譯器: GCC 11.1.0 編譯選項: -Wall -Wextra -std=c++17 -pedantic-errors https://wandbox.org/permlink/h1zB3mYSD3pCHiIV 使用以上編譯選項程式印出 9 次數字後就會正常結束 但是最佳化(-O2 或是 -O3)後執行會變成無窮迴圈 這是因為段程式碼有未定義行為(signed integer overflow) 由於編譯器在最佳化時可以假設 signed integer overflow 不會發生 編譯器會先將程式碼轉換為 1. for (int p = 0; p < 9 * 0x20000001; p += 0x20000001) { 2. std::cout << p << std::endl; 3. if (buf[0] == 'x') break; 4. } 接著把" p < 9 * 0x20000001 " 簡化為 true 因為 4,831,838,217 (9*0x20000001) 比 p 可能的最大值 INT_MAX 還大 所以結果永遠為 true 程式碼因此變成 1. for (int p = 0; true; p+=0x20000001) { 2. std::cout << p << std::endl; 3. if (buf[0] == 'x') break; 4. } 而形成無窮迴圈 -- https://i.imgur.com/kCK9K0T.jpg --



※ 發信站: 批踢踢實業坊(ptt.cc), 來自: 125.228.71.204 (臺灣)
※ 文章網址: https://webptt.com/m.aspx?n=bbs/C_and_CPP/M.1650944021.A.3AA.html
1F:→ Lipraxde: 原來 loop 優化會這樣做喔!(◎_◎;) 04/26 12:31
2F:推 descent: 感謝分享 04/26 14:55
3F:推 pinefruit: 有趣!感謝分享~ 04/27 00:54
4F:推 milkdragon: 推~~感謝分享 04/28 08:13
5F:推 CaptainH: 一直很不理解這種"優化"道理何在 04/28 10:01
6F:→ longlongint: 用常數先算好 把乘法拔掉而已 04/28 10:33
7F:→ longlongint: 應該先講拔常數乘法 再講overflow 會比較順? 04/28 10:34
8F:→ longlongint: 原po的寫法會讓人覺得編譯器在衝三小XD 04/28 10:34
9F:→ nh60211as: 我也覺得怪怪的,可是如果直接把p < 9*0x20000001 04/28 10:42
10F:→ nh60211as: 寫在 loop 條件裡編譯器會報 -Woverflow 警告, 04/28 10:44
11F:→ nh60211as: 所以它最佳化的方法滿有它在銃三小的感覺 04/28 10:44
12F:→ longlongint: 真的是衝三小耶 04/28 19:51
13F:→ tinlans: 這是 -faggressive-loop-optimizations 在 cunroll 做的 04/29 08:29
14F:→ tinlans: 事情,其實不用 -O3,用 -O1 編譯就看得見了,還會有警告 04/29 08:30
15F:→ tinlans: ,有興趣的可以編譯時下 -fdump-tree-all-details 然後看 04/29 08:32
16F:→ tinlans: 產生出來的 <file>.169t.cunroll 在做什麼,參考它前一步 04/29 08:32
17F:→ tinlans: <file>.156t.copyprop3 的輸出做前後比對。 04/29 08:33
18F:推 suhorng: @CaptainH 應該是典型的 induction variable elimination 05/03 15:50
19F:→ suhorng: 跟其他最佳化互動造成的意外 05/03 15:50
20F:→ suhorng: 畢竟程式原本沒有 undefined behavior 的話, 那換不換都 05/03 15:52
21F:→ suhorng: 不應該造成意外的結果. 所以產生意外的互動只能怪程式 05/03 15:52
22F:推 wulouise: 有UB就是程式有問題,所以會有什麼結果都不奇怪 05/08 20:01
23F:推 OnlyRD: 所以都要 cpplint & cppcheck 過才能上,用cppchec 05/15 20:31
24F:→ OnlyRD: k就抓出來了。 05/15 20:31
25F:推 Killercat: 不知道這例子 把p宣告為volatile int有沒有幫助 05/25 10:55
26F:推 LPH66: volatile 管的東西跟這個完全無關, 這並不是存取最佳化 05/25 18:18
27F:→ LPH66: 而是計算上的最佳化, 加上一些因為 UB 而有的假設出現的 05/25 18:19
28F:→ LPH66: 再提一次: UB = 編譯器可以方便行事, 假設不會有 UB 05/25 18:19







like.gif 您可能會有興趣的文章
icon.png[問題/行為] 貓晚上進房間會不會有憋尿問題
icon.pngRe: [閒聊] 選了錯誤的女孩成為魔法少女 XDDDDDDDDDD
icon.png[正妹] 瑞典 一張
icon.png[心得] EMS高領長版毛衣.墨小樓MC1002
icon.png[分享] 丹龍隔熱紙GE55+33+22
icon.png[問題] 清洗洗衣機
icon.png[尋物] 窗台下的空間
icon.png[閒聊] 双極の女神1 木魔爵
icon.png[售車] 新竹 1997 march 1297cc 白色 四門
icon.png[討論] 能從照片感受到攝影者心情嗎
icon.png[狂賀] 賀賀賀賀 賀!島村卯月!總選舉NO.1
icon.png[難過] 羨慕白皮膚的女生
icon.png閱讀文章
icon.png[黑特]
icon.png[問題] SBK S1安裝於安全帽位置
icon.png[分享] 舊woo100絕版開箱!!
icon.pngRe: [無言] 關於小包衛生紙
icon.png[開箱] E5-2683V3 RX480Strix 快睿C1 簡單測試
icon.png[心得] 蒼の海賊龍 地獄 執行者16PT
icon.png[售車] 1999年Virage iO 1.8EXi
icon.png[心得] 挑戰33 LV10 獅子座pt solo
icon.png[閒聊] 手把手教你不被桶之新手主購教學
icon.png[分享] Civic Type R 量產版官方照無預警流出
icon.png[售車] Golf 4 2.0 銀色 自排
icon.png[出售] Graco提籃汽座(有底座)2000元誠可議
icon.png[問題] 請問補牙材質掉了還能再補嗎?(台中半年內
icon.png[問題] 44th 單曲 生寫竟然都給重複的啊啊!
icon.png[心得] 華南紅卡/icash 核卡
icon.png[問題] 拔牙矯正這樣正常嗎
icon.png[贈送] 老莫高業 初業 102年版
icon.png[情報] 三大行動支付 本季掀戰火
icon.png[寶寶] 博客來Amos水蠟筆5/1特價五折
icon.pngRe: [心得] 新鮮人一些面試分享
icon.png[心得] 蒼の海賊龍 地獄 麒麟25PT
icon.pngRe: [閒聊] (君の名は。雷慎入) 君名二創漫畫翻譯
icon.pngRe: [閒聊] OGN中場影片:失蹤人口局 (英文字幕)
icon.png[問題] 台灣大哥大4G訊號差
icon.png[出售] [全國]全新千尋侘草LED燈, 水草

請輸入看板名稱,例如:Gossiping站內搜尋

TOP