c++ - 如何在编译时检查表达式是否非法?

标签 c++ sfinae typetraits c++-concepts

我的应用程序中有一个问题,我想断言一个函数应用程序将被编译器拒绝。有没有办法通过 SFINAE 进行检查?

例如,假设我想验证 std::transformconst 范围是非法的。到目前为止,这是我所拥有的:

#include <algorithm>
#include <functional>
#include <iostream>

namespace ns
{

using std::transform;

template<typename Iterator1, typename Iterator2, typename UnaryFunction>
  struct valid_transform
{
  static Iterator1 first1, last1;
  static Iterator2 first2;
  static UnaryFunction f;

  typedef Iterator2                   yes_type;
  typedef struct {yes_type array[2];} no_type;

  static no_type transform(...);

  static bool const value = sizeof(transform(first1, last1, first2, f)) == sizeof(yes_type);
};

}

int main()
{
  typedef int *iter1;
  typedef const int *iter2;
  typedef std::negate<int> func;

  std::cout << "valid transform compiles: " << ns::valid_transform<iter1,iter1,func>::value << std::endl;

  std::cout << "invalid transform compiles: " << ns::valid_transform<iter1,iter2,func>::value << std::endl;

  return 0;
}

不幸的是,我的特质拒绝合法和非法的情况。结果:

$ g++ valid_transform.cpp 
$ ./a.out 
valid transform compiles: 0
invalid transform compiles: 0

最佳答案

您的问题类似于 SFINAE + sizeof = detect if expression compiles .

该答案的

总结:sizeof 评估传递给它的表达式的类型,包括实例化函数模板,但它不会生成函数调用。这就是 Lol4t0 观察到 sizeof(std::transform(iter1(), iter1(), iter2(), func())) 编译的原因,即使 std::transform( iter1(), iter1(), iter2(), func()) 没有。

您的具体问题可以通过评估 Lol4t0 的答案中的模板来解决要提供给 std::transform 的任何输出范围。然而,使用 sizeof + SFINAE 技巧似乎无法解决在模板中验证函数调用是否会编译的一般问题。 (它需要一个可从运行时函数调用派生的编译时表达式)。

您不妨试试ConceptGCC看看这是否允许您以更方便的方式表达必要的编译时检查。

关于c++ - 如何在编译时检查表达式是否非法?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/10457804/

相关文章:

c++ - 无法使用 CMake 构建 STXXL

c++ - 将自定义类插入 unordered_map C++

c++ - 编译时动态函数调用

c++ - 如何在 C++ 中显示井字游戏的赢家?

c++ - MSVC 2013 'type': 不是 'std::enable_if<false,void> 的成员

C++11 元编程 : check for method existence

c++ - 检查类型是否在可变参数模板参数包中传递

c++ - is_convertible 用于多个参数

c++ - 我可以在编译时检查成员函数是否是运算符吗?

c++ - 通过指针运算访问结构体数据成员