C_and_CPP 板


LINE

※ 引述《lovejomi (JOMI)》之銘言: : https://wandbox.org/permlink/skxmougiYnaw1f1a : 一開始遭遇到這個compile warning : warning: instantiation of variable 'Foo<int>::bar' required here, but no defin : ition is available [-Wundefined-var-template] : 其實我不太知道他為什麼會warning : 我確實有define在test.cpp這個translation unit : 不懂的是 他感覺找不到定義 卻讓我pass, run time 也有拿到對的數值. : 而g++不會有warning : 1. 到這邊我還是不知道到底哪裡寫的不夠正確? : 然而我試著解決這warning : 於是我把 上面的 : //b. : // extern template class Foo<int>; : 打開 : https://wandbox.org/permlink/j2ANWRBoeIZZJAHE : link error : undefined reference to `Foo<int>::Test()' : 如果不呼叫 這個warning確實可以這樣解決 : 但我必須呼叫這function : 2. 為什麼他這樣會說undefined? .h裡面明確有包含定義阿? 先釐清兩件事: 1.你沒給定型態的template它就不是明確定義,自然它也不會被編譯。 2.template在兩份不同的.cpp檔裡給定型態,會在編譯時各自生成兩份實作。 回到你的code上面,當你使用extern template的時候 代表你指定在編譯時這份cpp檔參考其他cpp或.o裡的實作 而你在test.cpp裡的實作只有對bar做模板特化,那麼在編譯時test.cpp只會有bar有定義 其他symbol它不會自動幫你生,所以你的主程式自然找不到test()的實作 如果是使用template class Foo<int>的情況 你這個時候就是讓主程式生自己的實作了,而且因為它不是做特化而是直接給參 所以會整個class都生給你 : 然而 : //a. : // template class Foo<int>; : 打開後 : 3. 我認為我已經明確讓他產生程式碼了.... warning還是存在 : 回歸1. 的問題 我到底少做了什麼讓clang這樣出warning 殘念的是我這邊的clang還在3.8.1,好像跑不出這個warning(汗) 所以以下是我猜的,有錯歡迎版上高手指正XD 上面說過template會在不同的cpp生成各自的實作 你在test.cpp做特化時bar是有defination的,所以會替bar在.data留一個位子 到編主程式的時候它又為主程式生一份Foo<int>的實作, 這邊第二份實作的bar沒有給值,所以它是declaration 可是你對bar的defination必須等到做link的時候才會看到 compile階段編譯器是看不到的,所以compiler丟這個警告給你 提醒你bar這個static variable還是undefined的狀態 所以你link時有抓到它的symbol,應該就沒事這樣XD : 4. 對於template class內 有static function or data : 最正確的寫法該怎麼寫. : 網路上有查到 : 在test.h 直接寫 : template<class T> : int Foo<T>::bar = 初始直; : test.cpp一樣寫 : template<> : int Foo<int>::bar = 123; : 但我實際上在專案遇到一個況狀是 : 我某個cpp 寫Foo<int>::bar 拿到的卻是.h給的初始直(我認為是他初始化順序優先於te : st.cpp) : 所以目前毫無辦法處理這warning template最正確的寫法,就是通通給它塞在header裡 如果你覺得header有夠長,整個給你塞template塞得不要不要的 那就另外寫個.tcc檔,然後叫header把tcc include進來 要在.cpp用的時候就是只做給定型態的事,以你的例子來講 template<class T> int Foo<T>::bar = 0; template<> int Foo<int>::bar = 123; 這兩個就別再外寫cpp編譯了,直接放在test.h裡就好 不然就是再寫個test.tcc,然後在test.h最後面加一行#include "test.tcc" 以我自己的經驗,不管是boost還是標準庫基本上都是按這個模式在寫template : 5. 這似乎沒辦法用static是internal linkage 來解釋...讓我整個無法通透理解 : 請教各位有什麼方法處理這問題 : 謝謝 --



※ 發信站: 批踢踢實業坊(ptt.cc), 來自: 123.193.54.11
※ 文章網址: https://webptt.com/m.aspx?n=bbs/C_and_CPP/M.1529684256.A.902.html ※ 編輯: sarafciel (123.193.54.11), 06/23/2018 00:23:56
1F:推 lovejomi: 最後的說法 把Foo<int>::bar=123寫在header,兩個cpp inc 06/23 02:00
2F:→ lovejomi: lude就redefine了耶 06/23 02:00
3F:→ sarafciel: 汗 我寫一寫忘記把test.cpp補回去編譯orz 06/23 09:02
4F:→ sarafciel: 不過這個可以拿弱符號解掉的樣子 06/23 09:05
5F:→ sarafciel: int __attribute__((weak)) Foo<int>::bar =123; 06/23 09:06
6F:→ sarafciel: 把那行特化換成這個 06/23 09:07







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燈, 水草

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

TOP