- android - RelativeLayout 背景可绘制重叠内容
- android - 如何链接 cpufeatures lib 以获取 native android 库?
- java - OnItemClickListener 不起作用,但 OnLongItemClickListener 在自定义 ListView 中起作用
- java - Android 文件转字符串
我容器的一个构造函数默认构造一个分配器作为默认参数值:
template<class T, class Allocator>
struct my_container
{
my_container(int n, Allocator alloc = Allocator()) {}
};
据推测,只有当 Allocator
可以默认构造时,此构造函数才会启用。
我想用 std::is_constructible
测试此构造函数是否可以与不是默认可构造的分配器一起使用:
template<class T>
struct my_not_default_constructible_allocator
{
// no default ctor
my_not_default_constructible_allocator(int) {}
};
但是,当我应用 std::is_constructible
时,我得到一个编译时错误,而不是 false
,这正是我所期望的:
#include <type_traits>
template<class T, class Allocator>
struct my_container
{
my_container(int n, Allocator alloc = Allocator()) {}
};
template<class T>
struct my_not_default_constructible_allocator
{
// no default ctor
my_not_default_constructible_allocator(int) {}
};
int main()
{
bool result = std::is_constructible<my_container<int, my_not_default_constructible_allocator<int>>, int>::value;
return 0;
}
编译器输出:
$ clang -std=c++14 repro.cpp
repro.cpp:6:41: error: no matching constructor for initialization of 'my_not_default_constructible_allocator<int>'
my_container(int n, Allocator alloc = Allocator()) {}
^
/usr/bin/../lib/gcc/x86_64-linux-gnu/7.2.0/../../../../include/c++/7.2.0/type_traits:976:24: note: in instantiation of default function argument expression for
'my_container<int, my_not_default_constructible_allocator<int> >' required here
= decltype(::new _Tp(declval<_Arg>()))>
^
/usr/bin/../lib/gcc/x86_64-linux-gnu/7.2.0/../../../../include/c++/7.2.0/type_traits:977:24: note: in instantiation of default argument for '__test<my_container<int,
my_not_default_constructible_allocator<int> >, int>' required here
static true_type __test(int);
^~~~~~~~~~~
/usr/bin/../lib/gcc/x86_64-linux-gnu/7.2.0/../../../../include/c++/7.2.0/type_traits:987:24: note: while substituting deduced template arguments into function template '__test' [with _Tp =
my_container<int, my_not_default_constructible_allocator<int> >, _Arg = int, $2 = (no value)]
typedef decltype(__test<_Tp, _Arg>(0)) type;
^
/usr/bin/../lib/gcc/x86_64-linux-gnu/7.2.0/../../../../include/c++/7.2.0/type_traits:144:14: note: in instantiation of template class 'std::__is_direct_constructible_impl<my_container<int,
my_not_default_constructible_allocator<int> >, int>' requested here
: public conditional<_B1::value, _B2, _B1>::type
^
/usr/bin/../lib/gcc/x86_64-linux-gnu/7.2.0/../../../../include/c++/7.2.0/type_traits:992:14: note: in instantiation of template class 'std::__and_<std::is_destructible<my_container<int,
my_not_default_constructible_allocator<int> > >, std::__is_direct_constructible_impl<my_container<int, my_not_default_constructible_allocator<int> >, int> >' requested here
: public __and_<is_destructible<_Tp>,
^
/usr/bin/../lib/gcc/x86_64-linux-gnu/7.2.0/../../../../include/c++/7.2.0/type_traits:1074:14: note: in instantiation of template class 'std::__is_direct_constructible_new_safe<my_container<int,
my_not_default_constructible_allocator<int> >, int>' requested here
: public conditional<is_reference<_Tp>::value,
^
/usr/bin/../lib/gcc/x86_64-linux-gnu/7.2.0/../../../../include/c++/7.2.0/type_traits:1082:14: note: in instantiation of template class 'std::__is_direct_constructible_new<my_container<int,
my_not_default_constructible_allocator<int> >, int>' requested here
: public __is_direct_constructible_new<_Tp, _Arg>::type
^
/usr/bin/../lib/gcc/x86_64-linux-gnu/7.2.0/../../../../include/c++/7.2.0/type_traits:1122:14: note: in instantiation of template class 'std::__is_direct_constructible<my_container<int,
my_not_default_constructible_allocator<int> >, int>' requested here
: public __is_direct_constructible<_Tp, _Arg>
^
/usr/bin/../lib/gcc/x86_64-linux-gnu/7.2.0/../../../../include/c++/7.2.0/type_traits:1133:14: note: in instantiation of template class 'std::__is_constructible_impl<my_container<int,
my_not_default_constructible_allocator<int> >, int>' requested here
: public __is_constructible_impl<_Tp, _Args...>::type
^
<snip>
编译器详细信息:
$ clang --version
clang version 4.0.1-6 (tags/RELEASE_401/final)
Target: x86_64-pc-linux-gnu
Thread model: posix
InstalledDir: /usr/bin
std::is_constructible
的实现会导致错误,而不是 SFINAE 移除感兴趣的构造函数。
my_container
的构造函数的实现是否有误?
最佳答案
默认参数初始化似乎在 my_container
初始化的直接上下文中, [meta.unary_prop]/8 :
The predicate condition for a template specialization
is_constructible<T, Args...>
shall be satisfied if and only if the following variable definition would be well-formed for some invented variable t:
T t(declval<Args>()...);
[ Note: These tokens are never interpreted as a function declaration. — end note ] Access checking is performed as if in a context unrelated to T and any of the Args. Only the validity of the immediate context of the variable initialization is considered. [ Note: The evaluation of the initialization can result in side effects such as the instantiation of class template specializations and function template specializations, the generation of implicitly-defined functions, and so on. Such side effects are not in the “immediate context” and can result in the program being ill-formed. — end note ]
根据 [expr.call]/7:
The initialization and destruction of each parameter occurs within the context of the calling function.
因此可以推断默认参数初始化发生在“直接上下文”中。我的意见是这不是很清楚,术语直接上下文没有正式定义。
另一方面,Clang 还认为默认函数参数初始化发生在初始化表达式的直接上下文中。例如,此代码使用 Clang 编译:
template<class T,class =void>
struct constructible:std::false_type{};
template<class T>
struct constructible<T,std::void_t<decltype(T{std::declval<int>()})>>:std::true_type{};
int main()
{
static_assert(!constructible<my_container<int, my_not_default_constructible_allocator<int>>, int>::value);
return 0;
}
因此我们可以安全地假设这是一个 Clang 错误。
关于c++ - 为什么不使用 std::is_constructible 编译?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/54099360/
最近我试图检测特定私有(private)构造函数的存在,但遇到了 std::is_constructible 仅检查直接上下文的问题,因此不会识别任何此类构造函数。经过一些研究,我确实看到了一个答案
我想从我可以用模板参数构造的列表中找到类型。在这个例子中: using my_type = typename get_constructible::type; my_type 必须是 std::str
我想检测一个类是否有显式转换运算符。我已尝试使用 is_constructible,但以下断言因 msvc 19.00.23506 而失败。 #include #include struct Fo
这个问题在这里已经有了答案: Why does is_constructible claim something is constructible when it isn't? (2 个答案) 关闭
是否 std::is_constructible工作如果 Arg1是一种可转换为 T 的有效单参数构造函数的类型?当类型具有非模板化转换运算符时它似乎可以工作,但如果转换运算符是模板化的则它不起作用(
以下程序在使用 GCC 4.7 和 clang 3.2 编译时,会产生“1”作为输出。 #include struct foo { template foo(T) {
我有以下代码: #include class A; int main() { std::cout ::value struct is_constructible; T and all ty
源自this话题。也与此有关 topic . 我的问题是为什么 std::is_constructible 在直接上下文中停止?我认为 std::is_constructible 的用户会期望它能够深
我容器的一个构造函数默认构造一个分配器作为默认参数值: template struct my_container { my_container(int n, Allocator alloc = A
我在玩弄模板并尝试实现以下助手。 first_constructible::type 这将返回第一种 Types可从 Args... 构建.第一个问题显然是在 struct 中有两个参数包,所以我将用
std::is_constructible 的规则是什么?处理私有(private)构造函数?给定以下代码: #include class Class { private: Class()
这个问题在这里已经有了答案: C++98/03 std::is_constructible implementation (4 个答案) 关闭 5 年前。 到目前为止,我在网上找不到任何 ELI5。
我的爱好库的基本组件必须与 C++98 和 C++11 编译器一起使用。为了学习和享受自己,我创建了几种类型支持功能的 C++98 实现(如 enable_if、conditional、is_same
std::is_constructible::value 的结果不一致.我对该标准的解释是它应该是错误的。然而,Clang,同时具有 libc++ 和 libstdc++*,给出了 true。 GCC
我在这里冒险,假设这是可能的,但我不太确定。基本上我正在寻找的是一种在编译时在使用默认构造函数或通过引用采用一个参数的构造函数之间切换的方法。 即 T* create() { return n
我目前正在研究一些模板的东西。但我有一个问题。我有一堂这样的课 class myobj{ public: int val; char single; string na
我目前正在研究一些模板的东西。但我有一个问题。我有一堂这样的课 class myobj{ public: int val; char single; string na
在评估 std::is_constructible 和 std::is_destructible 时,Clang 和 GCC 似乎不遵守 friend 声明。 关于`is_constructible,
std::is_constructible 的预期结果是什么?在具有私有(private)或 protected 析构函数的类型上? 例如,即使只有 friend 可以释放它,我仍然可以在堆上构造这样
我正在尝试禁用具有非 std::string 可构造类型的 ctor。我的第一次尝试是这样的: #include struct A { template ::value>::type>
我是一名优秀的程序员,十分优秀!