C++返回值類型後置(跟蹤返回值類型)

在泛型編程中,可能需要通過參數的運算來得到返回值的類型。考慮下面這個場景:

template <typename R, typename T, typename U>R add(T t, U u){    return t+u;}int a = 1; float b = 2.0;auto c = add<decltype(a + b)>(a, b);

我們並不關心 a+b 的類型是什麼,因此,只需要通過 decltype(a+b) 直接得到返回值類型即可。但是像上面這樣使用十分不方便,因爲外部其實並不知道參數之間應該如何運算,只有 add 函數才知道返回值應當如何推導。


那麼,在 add 函數的定義上能不能直接通過 decltype 拿到返回值呢?

template <typename T, typename U>decltype(t + u) add(T t, U u)  // error: t、u尚未定義{    return t + u;}

當然,直接像上面這樣寫是編譯不過的。因爲 t、u 在參數列表中,而 C++ 的返回值是前置語法,在返回值定義的時候參數變量還不存在。


可行的寫法如下:

template <typename T, typename U>decltype(T() + U()) add(T t, U u){    return t + u;}

考慮到 T、U 可能是沒有無參構造函數的類,正確的寫法應該是這樣:

template <typename T, typename U>decltype((*(T*)0) + (*(U*)0)) add(T t, U u){    return t + u;}

雖然成功地使用 decltype 完成了返回值的推導,但寫法過於晦澀,會大大增加 decltype 在返回值類型推導上的使用難度並降低代碼的可讀性。


因此,在 C++11 中增加了返回類型後置(trailing-return-type,又稱跟蹤返回類型)語法,將 decltype 和 auto 結合起來完成返回值類型的推導。

返回類型後置語法是通過 auto 和 decltype 結合起來使用的。上面的 add 函數,使用新的語法可以寫成:


template <typename T, typename U>auto add(T t, U u) -> decltype(t + u){    return t + u;}

爲了進一步說明這個語法,再看另一個例子:

int& foo(int& i);float foo(float& f);template <typename T>auto func(T& val) -> decltype(foo(val)){    return foo(val);}

如果說前一個例子中的 add 使用 C++98/03 的返回值寫法還勉強可以完成,那麼這個例子對於 C++ 而言就是不可能完成的任務了。


在這個例子中,使用 decltype 結合返回值後置語法很容易推導出了 foo(val) 可能出現的返回值類型,並將其用到了 func 上。

返回值類型後置語法,是爲了解決函數返回值類型依賴於參數而導致難以確定返回值類型的問題。有了這種語法以後,對返回值類型的推導就可以用清晰的方式(直接通過參數做運算)描述出來,而不需要像 C++98/03 那樣使用晦澀難懂的寫法。


圖片


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