C++ 11實(shí)現(xiàn)檢查是否存在特定的成員函數(shù)
問題提出
最近工作中遇到這樣一個(gè)需求:實(shí)現(xiàn)一個(gè)ToString函數(shù)將類型T轉(zhuǎn)換到字符串,如果類型T中含有同名方法ToString則直接調(diào)用。
這樣一個(gè)ToString實(shí)現(xiàn)可以使用std::enable_if來做到,但是這里的難點(diǎn)在于如何判斷類型T中存在這樣一個(gè)ToString方法,以便可以放入enable_if中做SFINAE。
檢查類中是否存在特定成員
相同的問題在知乎上有人提出過,@孫明琦的答案提供了一個(gè)用于檢測特定檢測子U在類型T下是否有效的檢測器is_detected_v。其中用到了一個(gè)C++17的std::void_t,考慮到目前C++17還沒得用,這個(gè)實(shí)現(xiàn)只作參考之用(事實(shí)上C++17自帶了一個(gè)這樣的檢測器,并不需要自己寫這樣的模板)。
經(jīng)人提醒,我參考了下標(biāo)準(zhǔn)庫在實(shí)現(xiàn)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被用來檢查是否存在一個(gè)swap函數(shù)接受T作為參數(shù),很有趣的是__test函數(shù),如果存在swap函數(shù)滿足條件,那么test(int)這個(gè)重載版本就會(huì)被選中。而如果不滿足條件,因?yàn)橥茖?dǎo)失敗就剩下了test(…)這個(gè)版本。通過這一手段,再設(shè)置下返回值分別為true和false,就實(shí)現(xiàn)了這樣的一個(gè)檢測過程。
按圖索驥,檢查是否存在成員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中是否存在該成員的計(jì)算結(jié)果。
檢測是否存在特定成員函數(shù)
但是上述代碼有個(gè)問題,如果類T中的ToString是個(gè)成員變量,上述檢測也會(huì)返回true。
解決這一問題的手段是去調(diào)用T::ToString,如果這個(gè)ToString可以被調(diào)用并能生成返回值,就認(rèn)為這是個(gè)成員函數(shù)(嚴(yán)謹(jǐn)?shù)闹v,這個(gè)過程是確認(rèn)T::ToString是callable的,但是callable的玩意不一定就是成員函數(shù),然而實(shí)際使用并不需要這樣細(xì)分)。
這里的另一個(gè)問題是,因?yàn)門oString是成員函數(shù),那么decltype(T::ToString())這種手段就行不通了,因?yàn)槌蓡T函數(shù)必須帶對象進(jìn)行調(diào)用。既然必須要一個(gè)對象,那么這里的解決方法就是用上declval來產(chǎn)生一個(gè)對象,再用decltype獲取返回值類型。
按照這個(gè)思路,驗(yàn)證過程被改動(dòng)成:
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(...);
};
這個(gè)升級版本除了能檢查是否存在成員函數(shù)ToString以外還對返回值做了限定,確保返回的是string。以此類推,還能檢查返回是否是u16string、u32string。
總結(jié)
以上就是這篇文章的全部內(nèi)容了,希望本文的內(nèi)容對大家的學(xué)習(xí)或者工作能帶來一定的幫助,如果有疑問大家可以留言交流,謝謝大家對我們的支持。
上一篇:C++ HLSL實(shí)現(xiàn)簡單的圖像處理功能
欄 目:C語言
下一篇:簡單談?wù)凜++ 頭文件系列之(algorithm)
本文標(biāo)題:C++ 11實(shí)現(xiàn)檢查是否存在特定的成員函數(shù)
本文地址:http://www.jygsgssxh.com/a1/Cyuyan/1749.html
您可能感興趣的文章
- 04-02c語言沒有round函數(shù) round c語言
- 01-10數(shù)據(jù)結(jié)構(gòu)課程設(shè)計(jì)-用棧實(shí)現(xiàn)表達(dá)式求值的方法詳解
- 01-10使用OpenGL實(shí)現(xiàn)3D立體顯示的程序代碼
- 01-10深入理解C++中常見的關(guān)鍵字含義
- 01-10求斐波那契(Fibonacci)數(shù)列通項(xiàng)的七種實(shí)現(xiàn)方法
- 01-10C語言 解決不用+、-、&#215;、&#247;數(shù)字運(yùn)算符做加法
- 01-10使用C++實(shí)現(xiàn)全排列算法的方法詳解
- 01-10c++中inline的用法分析
- 01-10用C++實(shí)現(xiàn)DBSCAN聚類算法
- 01-10深入全排列算法及其實(shí)現(xiàn)方法


閱讀排行
本欄相關(guān)
- 04-02c語言函數(shù)調(diào)用后清空內(nèi)存 c語言調(diào)用
- 04-02func函數(shù)+在C語言 func函數(shù)在c語言中
- 04-02c語言的正則匹配函數(shù) c語言正則表達(dá)
- 04-02c語言用函數(shù)寫分段 用c語言表示分段
- 04-02c語言中對數(shù)函數(shù)的表達(dá)式 c語言中對
- 04-02c語言編寫函數(shù)冒泡排序 c語言冒泡排
- 04-02c語言沒有round函數(shù) round c語言
- 04-02c語言分段函數(shù)怎么求 用c語言求分段
- 04-02C語言中怎么打出三角函數(shù) c語言中怎
- 04-02c語言調(diào)用函數(shù)求fibo C語言調(diào)用函數(shù)求
隨機(jī)閱讀
- 04-02jquery與jsp,用jquery
- 08-05dedecms(織夢)副欄目數(shù)量限制代碼修改
- 01-11ajax實(shí)現(xiàn)頁面的局部加載
- 01-10SublimeText編譯C開發(fā)環(huán)境設(shè)置
- 01-10使用C語言求解撲克牌的順子及n個(gè)骰子
- 01-10delphi制作wav文件的方法
- 08-05DEDE織夢data目錄下的sessions文件夾有什
- 01-10C#中split用法實(shí)例總結(jié)
- 08-05織夢dedecms什么時(shí)候用欄目交叉功能?
- 01-11Mac OSX 打開原生自帶讀寫NTFS功能(圖文


