C++17一个很冷门很有意思的新特性

160 篇文章 11 订阅
152 篇文章 20 订阅

最近发现了一个有意思的特性:void_t。

void_t是C++17引入的一个新特性,它的定义很简单(有些编译器的实现可能不是这样,但也大体类似):

template< class... >
using void_t = void;

看着它很简单,但它搭配SFINAE却可以在模板元编程中发挥巨大作用。

比如在编译期判断类是否有某个类型using:

template <class, class = std::void_t<>>
struct has_type : std::false_type {};

template <class T>
struct has_type<T, std::void_t<typename T::type>> : std::true_type {};

比如判断是否有某个成员:

template <class, class = std::void_t<>>
struct has_a_member : std::false_type {};

template <class T>
struct has_a_member<T, std::void_t<decltype(std::declval<T>().a)>> : std::true_type {};

比如判断某个类是否可迭代:

template <typename, typename = void>
constexpr bool is_iterable{};

template <typename T>
constexpr bool is_iterable<T, std::void_t<decltype(std::declval<T>().begin()), decltype(std::declval<T>().end())>> = true;

比如判断某个类是否有某个函数:

template <class T, class = void>
struct has_hello_func : std::false_type {};

template <class T>
struct has_hello_func<T, std::void_t<decltype(std::declval<T>().hello())>> : std::true_type {};

测试结果:

struct HasType {
  typedef int type;
};
struct NHasType {
  int hello;
};

struct Hasa {
  int a;
};
struct NHasa {
  int b;
};

struct HasHello {
  void hello();
};
struct NoHasHello {};

int main() {
  std::cout << has_type<HasType>::value << '\n';   // 1
  std::cout << has_type<NHasType>::value << '\n';  // 0

  std::cout << has_a_member<Hasa>::value << '\n';   // 1
  std::cout << has_a_member<NHasa>::value << '\n';  // 0

  std::cout << has_hello_func<HasHello>::value << '\n';    // 1
  std::cout << has_hello_func<NoHasHello>::value << '\n';  // 0

  std::cout << is_iterable<std::vector<double>> << '\n';  // 1
  std::cout << is_iterable<double> << '\n';               // 0
}

它的原理其实就是利用SFINAE和模板优先找特化去匹配的特性,估计大家应该看示例代码就能明白。

关注B站号: 小鱼快来啊,免费领取300G编程资料

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值