ham*_*els 5 c++ const universal-reference
我认为通用引用(T&&)应该采用任何类型的引用.但以下不起作用.
当我尝试在我正在编写的库中进行const-correct时,我遇到了这个问题.我是C++的新手,之前没见过这样的东西.
TEST.CPP:
enum Cv_qualifier {
constant,
non_const
};
template <Cv_qualifier Cv> class A;
template<>
class A<Cv_qualifier::constant> {
public:
template<Cv_qualifier Cv2>
void t(const A<Cv2>&& out) {}
};
template <>
class A<Cv_qualifier::non_const> {
public:
template<Cv_qualifier Cv2>
void t(const A<Cv2>&& out) {}
};
int main()
{
A<Cv_qualifier::non_const> a;
A<Cv_qualifier::constant> b;
a.t(b);
}
Run Code Online (Sandbox Code Playgroud)
错误(编译g++ test.cpp -std=c++11):
test.cpp: In function ‘int main()’:
test.cpp:24:10: error: cannot bind ‘A<(Cv_qualifier)0u>’ lvalue to ‘const A<(Cv_qualifier)0u>&&’
a.t(b);
^
test.cpp:17:10: note: initializing argument 1 of ‘void A<(Cv_qualifier)1u>::t(const A<Cv2>&&) [with Cv_qualifier Cv2 = (Cv_qualifier)0u]’
void t(const A<Cv2>&& out) {}
^
Run Code Online (Sandbox Code Playgroud)
顺便说一句,在实际程序中,class A它不拥有任何实际数据,并包含对实际持有数据的另一个类的引用.我希望这意味着我不能不断创造的间接/复制数据时,我允许成员函数t的class A接受临时对象.
通用引用或转发引用仅因参考折叠而发生.它的工作方式如下:
T&& & -> T&
T& && -> T&
T&& && -> T&&
Run Code Online (Sandbox Code Playgroud)
这样,当您T&&在模板函数中收到时,右值引用可以根据其类型折叠为其他类型的引用T.在任何其他情况下,当崩溃没有发生时,SomeType&&将保留SomeType&&并将成为右值参考.
话虽如此,如果您希望您的功能支持转发,您可以这样做:
template <Cv_qualifier Cv> struct A;
template<>
struct A<Cv_qualifier::constant> {
template<typename T>
void t(T&& out) {}
};
template <>
struct A<Cv_qualifier::non_const> {
template<typename T>
void t(T&& out) {}
};
Run Code Online (Sandbox Code Playgroud)
实际上,现在崩溃发生了.如果Cv_qualifier要从中提取值T,可以使自己成为执行此操作的类型特征:
template<typename>
struct CvValue;
template<Cv_qualifier cv>
struct CvValue<A<cv>> {
constexpr static Cv_qualifier value = cv;
};
Run Code Online (Sandbox Code Playgroud)
然后,在你的函数中t,你可以这样做:
// v----- This is a good practice to apply a constraint
template<typename T, std::void_t<decltype(CvValue<std::decay_t<T>>::value)>* = 0>
auto t(T&& out) {
constexpr auto cv = CvValue<std::decay_t<T>>::value;
// do whatever you want with cv
}
Run Code Online (Sandbox Code Playgroud)
如果你不能使用C++ 17 std::void_t,你可以像这样实现它:
template<typename...>
using void_t = void;
Run Code Online (Sandbox Code Playgroud)
但是,如果您只想测试if T是否为A<...>,请使用:
template<typename>
struct is_A : std::false_type {};
template<Cv_qualifier cv>
struct is_A<A<cv>> : std::true_type {};
Run Code Online (Sandbox Code Playgroud)
不要忘记,使用它std::decay_t:
template<typename T, std::enable_if_t<std::is_A<std::decay_t<T>>::value>* = 0>
void t(T&& out) {}
Run Code Online (Sandbox Code Playgroud)
| 归档时间: |
|
| 查看次数: |
3264 次 |
| 最近记录: |