[問題] the C++ programming language裡面的問題

看板C_and_CPP (C/C++)作者 (殺拉頂)時間6年前 (2019/07/02 01:28), 6年前編輯推噓3(3026)
留言29則, 4人參與, 6年前最新討論串1/1
大家好, 最近剛好翻到the C++ programming language,章節C.13.8.4 有關templates, namespace 跟 specialization的, 看不太懂他所要表達的意思. 首先呢, 上面提到: .......省略....... This reflects three obvious strategies an implementation can use for generating specializations: [1] Generate a specialization the first time a call is seen. [2] At the end of a translation unit, generate all specializations needed for that translation unit. [3] Once every translation unit of a program has been seen, generate all specializations needed for the program. All three strategies have strengths and weaknesses, and combinations of these strategies are also possible. ......省略....... 然後說: A program is illegal, if it is possible to construct two different meanings by choosing different points of instantiation or different contents of namespaces at different possible contexts for generating the specialization. 然後有一段範例碼: namespace N { class A{ /* ... */ }; char f(A,int) ; } template<class T, class T2> char g(T t, T2 t2) { return f(t,t2) ; } //error (alternative resolutions of f(t)) char c= g(N: :A() ,'a') ; //<---- specialization namespace N //add to namespace N (§8.2.9.3 { void f(A,char) ; } 下面是我的疑惑: 對於文字解說, 我的理解是如果想像自己是compiler, 如果檢查最前面那三種產生 specialization的方法後, 發現產生的程式碼(例如像範例碼中的呼叫f(...))不同 則是一個編譯錯誤, 這邊的用詞是 the program is illegal. 但問題是compiler好像不太可能用三種方法產生程式碼之後再繼續後面的compiling吧? (不管最後有沒有編譯錯誤) 實際跑了一下跟玩一下範例碼, 感覺就只是產生specialization那行, 實際上被放到 void f(A, char)被定義的namespace之後, 也就是說被放到所有dependent namespace N 的後面而已, 這樣的話compiler就能盡量找到所有可能適用的函數..... 不過用這個解釋就不知道最一開始提的那三點是要幹嘛用的啦? 看的有點糊塗 在此請各位先進指點一下~~~感謝~~~ 其中一個疑惑就是那三點到底想表達的意思是什麼呢? P.S. 如果把 void f(A,char) 改成 char f(A,char) 就沒編譯錯誤了 可是場景還是相同阿? -- ※ 發信站: 批踢踢實業坊(ptt.cc), 來自: 36.224.44.2 (臺灣) ※ 文章網址: https://www.ptt.cc/bbs/C_and_CPP/M.1562002106.A.976.html ※ 編輯: saladim (36.224.44.2 臺灣), 07/02/2019 01:29:31 ※ 編輯: saladim (36.224.44.2 臺灣), 07/02/2019 01:31:38 ※ 編輯: saladim (36.224.44.2 臺灣), 07/02/2019 01:32:08

07/02 04:58, 6年前 , 1F
就只是告訴你編譯器可能用的演算法,再告訴你不論編譯器
07/02 04:58, 1F

07/02 04:58, 6年前 , 2F
用的是哪種編出來的一定都是 legal 的程式,然後講完什
07/02 04:58, 2F

07/02 04:58, 6年前 , 3F
麼是 illegal 之後舉個例給你看
07/02 04:58, 3F

07/02 23:11, 6年前 , 4F
可是這樣的話 如同在P.S.裡面所說的 改成 char f(A,char)
07/02 23:11, 4F

07/02 23:12, 6年前 , 5F
之後 specilazation產生在不同位置 可能會呼叫到不同函數
07/02 23:12, 5F

07/02 23:13, 6年前 , 6F
不是應該是illegal嗎? 但是實際上是可以編譯的...那不就
07/02 23:13, 6F

07/02 23:13, 6年前 , 7F
代表我使用的compiler其實是用混合策略?
07/02 23:13, 7F

07/02 23:49, 6年前 , 8F
不對 想一想還是覺得哪裡怪怪的 因為內文是說只要有可能
07/02 23:49, 8F

07/02 23:50, 6年前 , 9F
有不同意義(在不同位置產生碼) 那程式就屬不合法了....
07/02 23:50, 9F

07/03 02:28, 6年前 , 10F
因為overload resolution的緣故 你後面加的那個函式會
07/03 02:28, 10F

07/03 02:30, 6年前 , 11F
被優先選擇 然後會因為沒有回傳值編譯失敗
07/03 02:30, 11F

07/03 02:31, 6年前 , 12F
所以你把void改char當然就編得過了
07/03 02:31, 12F

07/03 07:45, 6年前 , 13F
欸... 不論用哪種策略都不會影響編譯成功與否吧,只是有
07/03 07:45, 13F

07/03 07:45, 6年前 , 14F
各自的優缺點(他這樣講我猜是最佳化難易度之類的編譯時
07/03 07:45, 14F

07/03 07:45, 6年前 , 15F
期才有差的優缺點)
07/03 07:45, 15F

07/03 07:45, 6年前 , 16F
你的 template 嘗試去 instantiate 一個 char f(A, char)
07/03 07:45, 16F

07/03 07:45, 6年前 , 17F
,但原本就已經有一個 void f(A, char) 了,如果你的編
07/03 07:45, 17F

07/03 07:45, 6年前 , 18F
譯器允許這個 instantiation 那就會造成有兩個不同版本
07/03 07:45, 18F

07/03 07:45, 6年前 , 19F
的 f(A, char),顯然你的編譯器不允許,所以才告訴你編
07/03 07:45, 19F

07/03 07:45, 6年前 , 20F
譯錯誤
07/03 07:45, 20F

07/03 07:45, 6年前 , 21F
但我想文說的兩個不同版本指的也可以是型別完全相同的兩
07/03 07:45, 21F

07/03 07:45, 6年前 , 22F
個不同 instantiation,但除非你的編譯器有 bug 否則沒
07/03 07:45, 22F

07/03 07:45, 6年前 , 23F
辦法藉由改變程式碼來造成這樣的結果,所以才用這種相似
07/03 07:45, 23F

07/03 07:45, 6年前 , 24F
的 example 給你看
07/03 07:45, 24F

07/03 07:47, 6年前 , 25F
* 兩個型別完全相同的 instance
07/03 07:47, 25F

07/03 09:12, 6年前 , 26F
單看這段節錄前後兩段沒直接關係吧?前者在說compiler
07/03 09:12, 26F

07/03 09:13, 6年前 , 27F
做specialisation 的策略,後者在說:吶,如果不同
07/03 09:13, 27F

07/03 09:13, 6年前 , 28F
位置就產生不同specialisation 的話那就是不合法的cod
07/03 09:13, 28F

07/03 09:13, 6年前 , 29F
e。這與選前一段哪種策略無關。
07/03 09:13, 29F
文章代碼(AID): #1T6aAwbs (C_and_CPP)
文章代碼(AID): #1T6aAwbs (C_and_CPP)