gpt4 book ai didi

c++ - 条件类型特征通用引用的问题

转载 作者:太空狗 更新时间:2023-10-29 21:18:58 25 4
gpt4 key购买 nike

所以我有一个函数,用于检查值并在值无效时抛出异常,否则将收到的值传回。我试图用普遍的崇敬和类型特征来概括这个例程。我觉得我很接近,因为我的例子在某些情况下有效,但不是全部。它似乎只适用于右值。

#include <iostream>
#include <utility>
#include <type_traits>
#include <string>
#include <vector>

using namespace std;

struct error_type{};

template <class T>
class has_empty
{
template<class U, class = typename std::enable_if<std::is_member_pointer<decltype(&U::empty)>::value>::type>
static std::true_type check(int);
template <class>
static std::false_type check(...);
public:
static constexpr bool value = decltype(check<T>(0))::value;
};

template <bool invalid>
struct valid {};


template <>
struct valid<false>
{
template<typename U, typename E>
static inline U&& check(U&& toCheck, const E& toThrow)
{
if (!toCheck)
std::cout << "NoEmpty Throw" << '\n';
else
std::cout << "NoEmpty valid" << '\n';
return std::forward<U>(toCheck);
}
};

template <>
struct valid<true>
{
template<typename U, typename E>
static inline U&& check(U&& toCheck, const E& toThrow)
{
if (toCheck.empty())
std::cout << "HasEmpty Throw" << '\n';
else
std::cout << "HasEmpty valid" << '\n';
return std::forward<U>(toCheck);
}
};

template<typename T
, typename E
, typename = typename std::enable_if<std::is_base_of<error_type, E>::value>::type>
inline T&& do_check(T&& toCheck, const E& toThrow)
{
return valid<has_empty<T>::value>::check(std::forward<T>(toCheck), toThrow);
}

struct HasEmpty
{
bool empty() {return false;}
};

struct NoEmpty
{
};


int main()
{
error_type e;

cout << has_empty<std::wstring>::value << '\n';
cout << has_empty<std::vector<std::wstring>>::value << '\n';
cout << has_empty<int>::value << '\n';
cout << has_empty<HasEmpty>::value << '\n';
cout << has_empty<NoEmpty>::value << '\n';

do_check(true, e);
do_check(false, e);

do_check(std::string("45"), e);
do_check(HasEmpty(), e);
do_check(std::vector<bool>(), e);
HasEmpty he;
do_check(std::move(he), e);
//do_check(he, e); // does not work, has_empty<T>::value is 0
}

产生输出

1
1
0
1
0
NoEmpty valid
NoEmpty Throw
HasEmpty valid
HasEmpty valid
HasEmpty Throw
HasEmpty valid

如果我取消注释最后一行,我会收到以下错误:

prog.cpp: In instantiation of 'static T&& valid<false, T, E>::check(T&&, const E&) [with T = HasEmpty&; E = error_type]':
prog.cpp:56:84: required from 'T&& do_check(T&&, const E&) [with T = HasEmpty&; E = error_type; <template-parameter-1-3> = void]'
prog.cpp:87:19: required from here
prog.cpp:30:11: error: no match for 'operator!' (operand type is 'HasEmpty')
if (!toCheck)
^
prog.cpp:30:11: note: candidate is:
prog.cpp:30:11: note: operator!(bool) <built-in>
prog.cpp:30:11: note: no known conversion for argument 1 from 'HasEmpty' to 'bool'

这似乎是has_empty<T>::value正在评估 false .我确信我可以做不同的工作来让它工作,所以在这一点上它有点学术性。不过,我们将不胜感激。

最佳答案

当您将左值传递给 do_check 时,它 deduces T as HasEmpty&在你的例子中。当然,引用类型没有名为 empty 的成员函数,并且表达式 decltype(&U::empty) 格式不正确,导致 static std::true_type check(int) 重载以获取 SFINAE。

如果你更换

static constexpr bool value = decltype(check<T>(0))::value;

static constexpr bool value = decltype(check<typename std::remove_reference<T>::type>(0))::value;

因此 U 永远不是引用类型,您的代码按预期工作。

Live demo

关于c++ - 条件类型特征通用引用的问题,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/28708516/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com