- c - 在位数组中找到第一个零
- linux - Unix 显示有关匹配两种模式之一的文件的信息
- 正则表达式替换多个文件
- linux - 隐藏来自 xtrace 的命令
背景:我正在编写类似 Either<A, B>
的包装器类型,我想要 return {some, args};
从返回 Either<A, B>
的函数开始工作函数返回 A
的确切时间或 B
.但是,我还想检测何时两者 A
和 B
可以用 {some, args}
初始化,并产生错误以保护用户免受歧义。
检测一个类型是否为T
可以从一些参数初始化,我试着写一个这样的函数:
template<typename T, typename... Args>
auto testInit(Args&&... args) -> decltype(T{std::forward<Args>(args)...});
// imagine some other fallback overload here...
我认为表达式 testInit<T>(some, args)
应该在 T{some, args}
时有效有效——在下面的代码中,初始化 auto x = MyType{1UL, 'a'};
有效,并且此测试也通过了:
struct MyType {
MyType(size_t a, char b) {}
};
auto x = MyType{1UL, 'a'}; // ok
static_assert(std::is_same<MyType, decltype(testInit<MyType>(1UL, 'a'))>::value, ""); // ok
但是,当我们从 std::initializer_list<char>
添加构造函数时,它打破了:
struct MyType {
MyType(size_t a, char b) {}
MyType(std::initializer_list<char> x) {} // new!
};
auto x = MyType{1UL, 'a'}; // still ok
// FAILS:
static_assert(std::is_same<MyType, decltype(testInit<MyType>(1UL, 'a'))>::value, "");
note: candidate template ignored: substitution failure [with T = MyType, Args = <unsigned long, char>]: non-constant-expression cannot be narrowed from type 'unsigned long' to 'char' in initializer list
auto testInit(Args&&... args) -> decltype(T{std::forward<Args>(args)...});
^ ~~~
为什么 Clang 拒绝解析我的 (size_t, char)
支持 initializer_list
的构造函数构造函数? 如何正确检测是否return {some, args};
将在返回 T
的函数中工作,无论它是聚合类型、用户定义的构造函数还是 initializer_list
构造函数?
最佳答案
有点复杂
而且我并不是真正的专家,所以我可以说一些不完全准确的话:请对我所说的持保留意见。
首先:当你写的时候
auto x = MyType{1UL, 'a'}; // ok
调用的构造函数是初始化列表之一,而不是接收 std::size_t
的构造函数和 char
.
这是有效的,因为第一个值 1UL
是一个 unsigned long int
但是有一个值(注意:值)可以缩小到 char
.那就是:因为 1UL
有效。是适合 char
的值.
如果你尝试
auto y = MyType{1000UL, 'a'}; // ERROR!
你得到一个错误,因为 1000UL
不能缩小到 char
.即:1000UL
不适合 char
.
这也适用于 decltype()
:
decltype( char{1UL} ) ch1; // compile
decltype( char{1000UL} ) ch2; // ERROR
但是考虑这个函数
auto test (std::size_t s)
-> decltype( char{s} );
此函数立即给出编译错误。
你可以想:“但是如果将 1UL
传递给 test()
,decltype()
可以将 std::size_t
的值缩小为 char
”
问题在于 C 和 C++ 是强类型语言;如果你允许 test()
,工作,返回一个类型,当收到一些值 std::size_t
,您可以创建(通过 SFINAE)一个函数,该函数返回某些值的类型和另一种类型的另一种类型。从强类型语言的角度来看,这是 Not Acceptable 。
所以
auto test (std::size_t s)
-> decltype( char{s} );
仅在 decltype( char{s} )
时才可接受对于 s
的所有可能值都是可接受的.即:test()
Not Acceptable ,因为 std::size_t
可以容纳 1000UL
不适合 char
.
现在稍微改变一下:制作test()
一个模板函数
template <typename T>
auto test (T s)
-> decltype( char{s} );
现在test()
编译;因为有类型 T
所有值都可以缩小到 char
( T = char
,例如)。所以test()
, 模板化,本质上并没有错。
但是当你将它与 std::size_t
一起使用时
decltype( test(1UL) ) ch; // ERROR
你得到一个错误,因为 test()
不能接受 std::size_t
.既不是可以缩小到 char
的值.
这正是您的代码的问题。
你的 testInit()
template <typename T, typename... Args>
auto testInit(Args&&... args)
-> decltype(T{std::forward<Args>(args)...});
是可以接受的,因为有类型 T
和 Args...
这样T{std::forward<Args>(args)...}
是可以接受的(例如:T = int
和 Args... = int
)。
但是T = MyType
和 Args... = std::size_t, char
是 Not Acceptable ,因为使用的构造函数是初始化列表为 char
的构造函数和非全部std::size_t
值可以缩小到 char
.
结论:编译时出错 decltype(testInit<MyType>(1UL, 'a')
因为编译时出错 MyType{1000UL, 'a'}
.
奖励答案:我建议对您的 testInit()
进行改进(恕我直言) .
使用 SFINAE 和逗号运算符的强大功能,您可以编写
template <typename T, typename... Args>
auto testInit (Args ... args)
-> decltype( T{ args... }, std::true_type{} );
template <typename...>
std::false_type testInit (...);
所以你可以写一些static_assert()
简单如下
static_assert( true == decltype(testInit<MyType>('a', 'b'))::value, "!");
static_assert( false == decltype(testInit<MyType>(1UL, 'b'))::value, "!");
后记:如果你想要 MyType(size_t a, char b) {}
调用构造函数,可以使用圆括号
auto y = MyType(1000UL, 'a'); // compile!
所以如果你写 testInit()
带圆括号
template <typename T, typename... Args>
auto testInit (Args ... args)
-> decltype( T( args... ), std::true_type{} );
template <typename...>
std::false_type testInit (...);
您通过了以下两项 static_assert()
static_assert( true == decltype(testInit<MyType>('a', 'b'))::value, "!");
static_assert( true == decltype(testInit<MyType>(1UL, 'b'))::value, "!");
关于c++ - 如何检测类型是否可列表初始化?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/47343146/
#include using namespace std; class C{ private: int value; public: C(){ value = 0;
这个问题已经有答案了: What is the difference between char a[] = ?string?; and char *p = ?string?;? (8 个回答) 已关闭
关闭。此题需要details or clarity 。目前不接受答案。 想要改进这个问题吗?通过 editing this post 添加详细信息并澄清问题. 已关闭 7 年前。 此帖子已于 8 个月
除了调试之外,是否有任何针对 c、c++ 或 c# 的测试工具,其工作原理类似于将独立函数复制粘贴到某个文本框,然后在其他文本框中输入参数? 最佳答案 也许您会考虑单元测试。我推荐你谷歌测试和谷歌模拟
我想在第二台显示器中移动一个窗口 (HWND)。问题是我尝试了很多方法,例如将分辨率加倍或输入负值,但它永远无法将窗口放在我的第二台显示器上。 关于如何在 C/C++/c# 中执行此操作的任何线索 最
我正在寻找 C/C++/C## 中不同类型 DES 的现有实现。我的运行平台是Windows XP/Vista/7。 我正在尝试编写一个 C# 程序,它将使用 DES 算法进行加密和解密。我需要一些实
很难说出这里要问什么。这个问题模棱两可、含糊不清、不完整、过于宽泛或夸夸其谈,无法以目前的形式得到合理的回答。如需帮助澄清此问题以便重新打开,visit the help center . 关闭 1
有没有办法强制将另一个 窗口置于顶部? 不是应用程序的窗口,而是另一个已经在系统上运行的窗口。 (Windows, C/C++/C#) 最佳答案 SetWindowPos(that_window_ha
假设您可以在 C/C++ 或 Csharp 之间做出选择,并且您打算在 Windows 和 Linux 服务器上运行同一服务器的多个实例,那么构建套接字服务器应用程序的最明智选择是什么? 最佳答案 如
你们能告诉我它们之间的区别吗? 顺便问一下,有什么叫C++库或C库的吗? 最佳答案 C++ 标准库 和 C 标准库 是 C++ 和 C 标准定义的库,提供给 C++ 和 C 程序使用。那是那些词的共同
下面的测试代码,我将输出信息放在注释中。我使用的是 gcc 4.8.5 和 Centos 7.2。 #include #include class C { public:
很难说出这里问的是什么。这个问题是含糊的、模糊的、不完整的、过于宽泛的或修辞性的,无法以目前的形式得到合理的回答。如需帮助澄清此问题以便重新打开它,visit the help center 。 已关
我的客户将使用名为 annoucement 的结构/类与客户通信。我想我会用 C++ 编写服务器。会有很多不同的类继承annoucement。我的问题是通过网络将这些类发送给客户端 我想也许我应该使用
我在 C# 中有以下函数: public Matrix ConcatDescriptors(IList> descriptors) { int cols = descriptors[0].Co
我有一个项目要编写一个函数来对某些数据执行某些操作。我可以用 C/C++ 编写代码,但我不想与雇主共享该函数的代码。相反,我只想让他有权在他自己的代码中调用该函数。是否可以?我想到了这两种方法 - 在
我使用的是编写糟糕的第 3 方 (C/C++) Api。我从托管代码(C++/CLI)中使用它。有时会出现“访问冲突错误”。这使整个应用程序崩溃。我知道我无法处理这些错误[如果指针访问非法内存位置等,
关闭。这个问题不符合Stack Overflow guidelines .它目前不接受答案。 我们不允许提问寻求书籍、工具、软件库等的推荐。您可以编辑问题,以便用事实和引用来回答。 关闭 7 年前。
已关闭。此问题不符合Stack Overflow guidelines 。目前不接受答案。 要求我们推荐或查找工具、库或最喜欢的场外资源的问题对于 Stack Overflow 来说是偏离主题的,因为
我有一些 C 代码,将使用 P/Invoke 从 C# 调用。我正在尝试为这个 C 函数定义一个 C# 等效项。 SomeData* DoSomething(); struct SomeData {
这个问题已经有答案了: Why are these constructs using pre and post-increment undefined behavior? (14 个回答) 已关闭 6
我是一名优秀的程序员,十分优秀!