C_and_CPP 板


LINE

※ 引述《m8403051 (大吉嶺紅茶)》之銘言: : 開發平台(Platform): Linux CentOS 7 : 編譯器: gcc version 4.4.7 20120313 (Red Hat 4.4.7-23) : 預期的正確結果(Expected Output):180 : 錯誤結果(Wrong Output):215 : 程式碼(Code):(請善用置底文網頁, 記得排版,禁止使用圖檔) : https://ideone.com/e.js/nn576o : #include <iostream> : using namespace std; : int main() { : int a=6, b=7; : a *= a-- * --b; : cout << a << endl; : return 0; : } : 補充說明(Supplement): : 使用 ideone.com 跑出來是正確的結果 180, 但是使用 CentOS 7 g++ 跑出來卻是 215 : 請教各位先進這是哪邊沒注意到? GCC 4.4 算比較舊的編譯器, 預設會以 C++98 來編譯程式碼, 敘述 的求值 (evaluation) 就看標準定義的 sequence point 有幾個以 及插在什麼地方. 我手邊只有 ISO/IEC 14882:2003 所以可能和更 早的資料有點出入, C++98 和原文有關的描述主要如下: [expr] 5.4 https://i.imgur.com/9WB2V5H.png 剛好範例裡的 i = ++i + 1 因為 sequence point 在這個 statem- ant 裡只有一個, 即完整敘述的結束, 多次更改 i 的值所以行為是 unspecified; 但是這在 C++11 (N3242) 裡定義得更精確了: [intro.execution] 1.9/15 https://i.imgur.com/b8zsTdZ.png 因為有了 sequenced before 的觀念, 在等號右邊使用一次 prefix ++ 的行為被明確定義了: The value computations of the operands of an operator are sequenced before the value computation of the result of the operator 和 C++98 相比, 在等號右邊敘述計算 + 之前多了一個 sequence point, 所以 prefix ++ 的問題解了, 再來 postfix ++ 要到 C++17 (N4659) 才經由 [P0145] 確立執行結果: [P0145R3] [Core] Refining Expression Evaluation Order for Idiomatic C++ https://bit.ly/2z9o0vM [intro.execution] 4.6/17 https://i.imgur.com/9fmE9yJ.png [expr.ass] 8.18/1 https://i.imgur.com/K3nR6zk.png C++17 直接保證了等號左右邊的求值順序 (先右後左). 總結一下兩 種敘述在各年代標準底下的行為 (結果): ┌─────┬──────┬──────┐ │ Standard │i = --i + 1 │i = i-- + 1 │ ├─────┼──────┼──────┤ │ C++98 │ UBUB │ ├─────┼──────┼──────┤ │ C++11 │ iUB │ ├─────┼──────┼──────┤ │ C++17 │ ii + 1 │ └─────┴──────┴──────┘ 再回到原來的程式碼: int a = 6, b = 7; a *= ((a--) * (--b)); --b 基本上可以直接代換成 6, 變成只有 a 的運算: a *= ((a--) * 6); 那麼問題來了, 如果 *= 左邊和右邊的相對求值順序沒有定下來, 此題就沒標準答案 (與賦值無關), 但如果以 C++17 或更新的標準 來看: 等號右邊 (a--) 的 side effect 會先被產生, 然後才是等 號左邊拿 a 的值來做運算, 這時候就形同於以下的賦值敘述: a *= (6 * 6); // a becomes 5 after a-- => a = 5 * (6 * 6); // a = 180 -- P1389R0: Guidelines for Teaching C++ to Beginners https://bit.ly/2GvDWKb SG20 Education and Recommended Videos for Teaching C++ https://www.cjdb.com.au/sg20-and-videos --



※ 發信站: 批踢踢實業坊(ptt.cc), 來自: 123.193.76.216 (臺灣)
※ 文章網址: https://webptt.com/m.aspx?n=bbs/C_and_CPP/M.1587389865.A.CA4.html
1F:→ MOONRAKER: evaluate作「求值」或「計算」會好一點 就像javascript 04/21 08:58
2F:→ MOONRAKER: 或perl的eval()不叫評估函數 (雖然好像有人這樣翻) 04/21 08:59
感謝, 修正好了 :)
3F:推 m8403051: 謝謝, 所以更新 gcc 版本就能避免錯誤 04/21 09:52
4F:→ m8403051: 正確答案應該是 180 沒錯 04/21 09:52
更新後記得要用 -std=c++17 來指定標準版本 ※ 編輯: poyenc (61.216.75.43 臺灣), 04/21/2020 10:20:15







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