C++模板,判斷是否存在成員函數,實現差異化操作

工作中遇到一個問題,我有一個容器,裝着各式各樣的對象的指針,需要把擁有dump方法的指針內容dump出去,而對於沒有dump方法的對象,直接忽略。

首先想到的是給每個對象提供一個查詢操作,從而得知是否擁有dump方法。顯然這個方法不能讓人滿意,需要更改大量的class實現。C++如果我能自動判斷某個類型是否擁有某方法,這個問題可以完美的解決,因爲是否含有某方法編譯期已經確定了,所以是有可能通過一些技巧來實現這個功能的。

查閱了大量的模板偏特化,匹配失敗不是錯誤,終於找到了Mr.Right:
廢話不多說,貼代碼:
www.huaqiangu6.com

檢測是否有定義  void hello():

點擊(此處)摺疊或打開

  1. template<typename T>

  2. struct has_hello{

  3.     template<typename U, void (U::*)()> struct HELPS;

  4.     template<typename U> static char Test(HELPS<U, &U::hello>*);

  5.     template<typename U> static int Test(...);

  6.     const static bool Has = sizeof(Test<T>(0)) == sizeof(char);

  7. }

測試:

點擊(此處)摺疊或打開

  1. struct A

  2. {

  3.   void hello(){

  4.     cout<<"A is Hello."<<endl;

  5.   }

  6.   int x;

  7. };



  8. int main()

  9. {

  10.   cout<<"A has hello? "<<has_hello<A>::Has<<endl;

  11. }


 has_hello 能編譯通過,而且能工作!!!

解釋:
template<typename U> static int Test(...); 能匹配所有的類型U
template<typename U, void (U::*)()> struct HELPS當Type U有hello,且hello的類型爲 void (U::*)()時,模板HELPS<U, &U::hello>能正確匹配,否則模板無法匹配

Test<T>(0)  優先匹配 HELPS<U, &U::hello> 因爲template<typename U> static char Test(HELPS<U, &U::hello>*是一個特化

sizeof操作符不需要計算表達式的值,是一個編譯期的操作,定義指針類型可以只有聲明沒有定義
所以HELPS和Test都不需要實現,僅有聲明就可以通過編譯

巧妙的探測了自定義類型A是否含有void U::hello() 方法。


發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章