C++11實現檢查是否存在特定的成員函數

問題提出

創新互聯建站是網站建設技術企業,為成都企業提供專業的做網站、網站制作,網站設計,網站制作,網站改版等技術服務。擁有10余年豐富建站經驗和眾多成功案例,為您定制適合企業的網站。10余年品質,值得信賴!

最近工作中遇到這樣一個需求:實現一個ToString函數將類型T轉換到字符串,如果類型T中含有同名方法ToString則直接調用。

這樣一個ToString實現可以使用std::enable_if來做到,但是這里的難點在于如何判斷類型T中存在這樣一個ToString方法,以便可以放入enable_if中做SFINAE。

檢查類中是否存在特定成員

相同的問題在知乎上有人提出過,@孫明琦的答案提供了一個用于檢測特定檢測子U在類型T下是否有效的檢測器is_detected_v。其中用到了一個C++17的std::void_t,考慮到目前C++17還沒得用,這個實現只作參考之用(事實上C++17自帶了一個這樣的檢測器,并不需要自己寫這樣的模板)。

經人提醒,我參考了下標準庫在實現swap上做的努力,看到了這樣的寫法:

namespace __swappable_details {
 using std::swap;
 
 struct __do_is_swappable_impl
 {
  template <typename _Tp, typename
    = decltype(swap(std::declval<_Tp&>(), std::declval<_Tp&>()))>
  static true_type __test(int);
 
  template <typename>
  static false_type __test(...);
 };
}
 
template <typename _Tp>
struct __is_swappable_impl
 : public __swappable_details::__do_is_swappable_impl
{
 typedef decltype(__test<_Tp>(0)) type;
};
 
template <typename _Tp>
struct __is_swappable
 : public __is_swappable_impl<_Tp>::type
{};

簡單分析可以看到__is_swappable被用來檢查是否存在一個swap函數接受T作為參數,很有趣的是__test函數,如果存在swap函數滿足條件,那么test(int)這個重載版本就會被選中。而如果不滿足條件,因為推導失敗就剩下了test(…)這個版本。通過這一手段,再設置下返回值分別為truefalse,就實現了這樣的一個檢測過程。

按圖索驥,檢查是否存在成員ToString的模板就可以這么寫:

namespace details
{
 struct HasMemberToStringValidator
 {
  template <typename T, typename = decltype(&T::ToString)>
  static std::true_type Test(int);
 
  template <typename>
  static std::false_type Test(...);
 };
}
 
template <typename T>
struct HasMemberToString :
 public decltype(details::HasMemberToStringValidator::Test<T>(0))
{};

HasMemberToString::value就是T中是否存在該成員的計算結果。

檢測是否存在特定成員函數

但是上述代碼有個問題,如果類T中的ToString是個成員變量,上述檢測也會返回true。

解決這一問題的手段是去調用T::ToString,如果這個ToString可以被調用并能生成返回值,就認為這是個成員函數(嚴謹的講,這個過程是確認T::ToString是callable的,但是callable的玩意不一定就是成員函數,然而實際使用并不需要這樣細分)。

這里的另一個問題是,因為ToString是成員函數,那么decltype(T::ToString())這種手段就行不通了,因為成員函數必須帶對象進行調用。既然必須要一個對象,那么這里的解決方法就是用上declval來產生一個對象,再用decltype獲取返回值類型。

按照這個思路,驗證過程被改動成:

struct HasMemberToStringValidator
{
 template <typename T, typename U =
  typename std::decay<decltype(std::declval<T>().ToString())>::type,
  typename = typename std::enable_if<std::is_same<std::string, U>::value>::type>
 static std::true_type Test(int);
 
 template <typename>
 static std::false_type Test(...);
};

這個升級版本除了能檢查是否存在成員函數ToString以外還對返回值做了限定,確保返回的是string。以此類推,還能檢查返回是否是u16string、u32string。

總結

以上就是這篇文章的全部內容了,希望本文的內容對大家的學習或者工作能帶來一定的幫助,如果有疑問大家可以留言交流,謝謝大家對創新互聯的支持。

新聞標題:C++11實現檢查是否存在特定的成員函數
轉載源于:http://www.kartarina.com/article38/jchdsp.html

成都網站建設公司_創新互聯,為您提供做網站關鍵詞優化定制網站網站策劃用戶體驗網站營銷

廣告

聲明:本網站發布的內容(圖片、視頻和文字)以用戶投稿、用戶轉載內容為主,如果涉及侵權請盡快告知,我們將會在第一時間刪除。文章觀點不代表本網站立場,如需處理請聯系客服。電話:028-86922220;郵箱:631063699@qq.com。內容未經允許不得轉載,或轉載時需注明來源: 創新互聯

成都做網站
主站蜘蛛池模板: 无码中文字幕人妻在线一区二区三区 | 亚洲熟妇无码久久精品| 亚洲人成无码www久久久| 无码专区狠狠躁躁天天躁| MM1313亚洲精品无码久久| 无码专区狠狠躁躁天天躁 | 五月丁香六月综合缴清无码| 无码日韩人妻精品久久蜜桃| 久久久久成人精品无码| 无码狠狠躁久久久久久久| 无码一区二区三区免费| 18禁超污无遮挡无码免费网站| 无码h黄肉3d动漫在线观看| 亚洲av日韩av无码av| 人妻无码一区二区三区免费| 亚洲精品无码成人AAA片| 亚洲AV无码成H人在线观看| av中文无码乱人伦在线观看| 亚洲AV日韩AV永久无码色欲| 久久久久久亚洲Av无码精品专口| 亚洲日韩欧洲无码av夜夜摸| 久久久久亚洲AV无码专区桃色| 内射中出无码护士在线| 无码中文人妻在线一区二区三区| 久久无码AV一区二区三区| 精品无码AV一区二区三区不卡| 无码午夜成人1000部免费视频| 亚洲AV日韩AV永久无码免下载 | 无码国产精品一区二区免费式影视 | 成人无码一区二区三区| 亚洲国产成人精品无码久久久久久综合 | 亚洲中久无码永久在线观看同| 欧洲精品无码一区二区三区在线播放| 性色AV一区二区三区无码| 国产精品无码一区二区三区免费| 久久AV无码精品人妻出轨| 曰韩无码AV片免费播放不卡| 日韩人妻无码精品专区| 国产午夜无码精品免费看| 久久久无码精品亚洲日韩按摩| 亚洲色无码国产精品网站可下载|