- iOS/Objective-C 元类和类别
- objective-c - -1001 错误,当 NSURLSession 通过 httpproxy 和/etc/hosts
- java - 使用网络类获取 url 地址
- ios - 推送通知中不播放声音
我正在尝试创建一个示例类,它根据其模板参数是否可以用某些参数构造而具有不同的特化。在我的示例中,使用了一个简单的 int
。我试过:
template<class A_t>
struct creator
{
A_t* ptr = nullptr;
};
template<class A_t>
struct creator<decltype(A_t(int()))>
{
A_t* ptr = new A_t(5);
};
struct A
{
int i;
A(int i) : i(i) {}
};
int main() {
std::cout << creator<A>().ptr << std::endl;
return 0;
}
我的意图是它打印一个自动构造的对象的内存地址。但是,它打印的是 0。因此,它采用的是非专用模板。
A_t
可以用该语法推导出来,因为 A_t
是明确给出的。此外,decltype(A_t(int())
的类型是A_t
(例如不是A_t&&
),一个简单的测试:
std::cout << std::is_same<decltype(A(int()), A>::value << std::endl;
打印 1。
但是,该实现有效:
#include <iostream>
template<class A_t, class = A_t>
struct creator
{
A_t* ptr = nullptr;
};
template<class A_t>
struct creator<A_t, decltype(A_t(int()))>
{
A_t* ptr = new A_t(5);
};
struct A
{
int i;
A(int i) : i(i) {}
};
int main() {
std::cout << creator<A>().ptr << std::endl;
return 0;
}
Coliru用两个类进行测试,一个接受 int
作为参数,另一个不接受。
为什么第一种方法不起作用?
最佳答案
我对你问题背后的想法最好的解释是:
对于给定的实例化类型 U
=> A_t
, 如果转换 U(int)
未定义则替换 U(int)
=> A_t(int)
将在上下文中失败 decltype(A_t(int()))
和专业:
template<class A_t>
struct creator<decltype(A_t(int()))>
{
A_t* ptr = new A_t(5);
};
将被删除,只留下基础模板实例化。但如果转换 U(int)
定义为替换将成功。
那么,因为:
std::is_same<U,decltype(U(int()))>::value == true
两位候选人:
// Tweedledum, with of U => A_t
struct creator<U>
{
U* ptr = nullptr;
};
和:
// Tweedledee, with U => decltype(A_t(int()))
struct creator<U>
{
U* ptr = new U(5);
};
正在运行中。
然后,将选择最专业的候选人,这将是Tweedledee
,因为满足U
=> decltype(A_t(int()))
约束 U
比裸体U => A_t
.
这个推理隐含地依赖于它对于 Teedledee
是可推导的那U
= decltype(A_t(int()))
什么时候U
=> A_t
;归结为可以推断出 U(int)
= A_t(int)
在唯一的模板参数中 decltype(A_t(int()))
.
你相信是这样然后想知道 gcc 如何选择 Tweedledum
.
正如@WhozCraig 所指出的,clang++ 明确拒绝了您的可推导性前提:-
$ clang++-3.8 -Wall -Wextra -pedantic -std=c++14 main.cpp
main.cpp:10:8: warning: class template partial specialization contains a template parameter that cannot be deduced; this partial specialization will never be used
struct creator<decltype(A_t(int()))>
^~~~~~~~~~~~~~~~~~~~~~~~~~~~~
main.cpp:9:16: note: non-deducible template parameter 'A_t'
template<class A_t>
^
事实证明,gcc 在这里被引入了一个诊断错误你的 SFINAE 措辞不吉利,decltype(A_t(int()))
如果将其替换为 decltype(A_t{int()})
, 然后 gcc 也是 gets with the program :
$ g++-6 -Wall -Wextra -pedantic -std=c++14 main.cpp
main.cpp:10:8: error: template parameters not deducible in partial specialization:
struct creator<decltype(A_t{int()})>
^~~~~~~~~~~~~~~~~~~~~~~~~~~~~
main.cpp:10:8: note: ‘A_t’
在那之后,clang++ 是三个编译器中唯一的编译器完全编译程序 - 并实例化 creator<A>
来自基本模板。
最终的共识是A_t
在 decltype(A_t{int()})
中不可推导得到 C++14 标准的认可:
Deducing template arguments from a type [temp.deduct.type]
1 Template arguments can be deduced in several different contexts, but in each casea type that is specified in terms of template parameters (call it P) is comparedwith an actual type (call it A), and an attempt is made to find template argumentvalues (a type for a type parameter, a value for a non-type parameter, or atemplate for a template parameter) that will make P, after substitution of thededuced values (call it the deduced A), compatible with A.
...
4 In most cases, the types, templates, and non-type values that are used to composeP participate in template argument deduction... In certain contexts, however,the value does not participate in type deduction, but instead uses the valuesof template arguments that were either deduced elsewhere or explicitly specified.If a template parameter is used only in non-deduced contexts and is not explicitlyspecified, template argument deduction fails.
5 The non-deduced contexts are:
...
(5.2) - The expression of a decltype-specifier.
...
第 4 段还解释了为什么您的第二种方法会成功。
关于c++ - 构造函数参数的 SFINAE,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/41027701/
谁能解释一下原因: (define a (lambda() (cons a #f))) (car (a)) ==> procedure ((car (a))) ==> (procedure . #f)
这是 PyBrain 网站的摘录。我了解大部分正在发生的事情,但是一行让我完全难住了。我以前从未在 python 代码中看到过这样的东西。这是整个循环,对于上下文: for c in [0,
我是gradle / groovy的新手。我想创建将做一些事情的自定义任务。我的第一个问题是任务完成时该如何做?我可以覆盖doFirst / doLast闭包吗?也许我可以重写某些在开始和结束时都会执
我刚刚开始评估 MS 企业库。他们使用以下指令来获取实例: var customerDb = EnterpriseLibraryContainer.Current.GetInstance("C
这是我的 if else Ansible 逻辑.. - name: Check certs exist stat: path=/etc/letsencrypt/live/{{ rootDomain
我正在使用construct 2.8 对一些失传已久的 Pascal 程序创建的一些文件的 header 进行逆向工程。 header 由许多不同的记录组成,其中一些是可选的,我不确定顺序是否固定。
我在将 getchar() 的输入放入 char *arr[] 数组时遇到问题。我这样做的原因是因为输入数据(将是一个带有命令行参数的文件)将存储在一个 char 指针数组中以传递给 execvp 函
通常我们不能约束类型参数 T派生自密封类型(例如 struct 类型)。这将毫无意义,因为只有一种类型适合,因此不需要泛型。所以约束如下: where T : string 或: where T :
关闭。此题需要details or clarity 。目前不接受答案。 想要改进这个问题吗?通过 editing this post 添加详细信息并澄清问题. 已关闭 9 年前。 Improve th
#include using namespace std; class A { private: int m_i; friend int main(int argc, char cons
这个问题在这里已经有了答案: Are there legitimate uses for JavaScript's "with" statement? (33 个答案) 关闭 9 年前。 我有这个代
在this answer我看到了下一个 Bash 结构。 yes "$(< file.txt)" 什么意思 "$(< file.txt)" ? 我明白了 命令替换 - $(command)用命令的结
if (a == 1) //do something else if (a == 2) //do something else if (a == 3) //do somethi
关于构造的快速简单的问题。 我有以下用于将项目添加到 ListView 的代码。 ListViewItem item = new ListViewItem(); item.Text = file; i
我想使用 std::vector 来控制给定的内存。首先,我很确定这不是好的做法,但好奇心占了上风,无论如何我都想知道如何做到这一点。 我遇到的问题是这样的方法: vector getRow(unsi
下面显示了一段简单的javascript: var mystring = ("random","ignored","text","h") + ("ello world") 这个字符串会生成 hello
在 Java 中,创建对象的标准方法是使用 MyClass name = new MyClass(); 我也经常看到构造 new MyClass() { /*stuff goes in here*/
我正在编写 C++ ndarray 类。我需要动态大小和编译时大小已知的数组(分别分配自由存储和分配堆栈)。我想支持从嵌套的 std::initializer_list 进行初始化。 动态大小的没问题
我正在将一个项目从 Visual Studio 2005 转换为 Visual Studio 2008,并提出了上述结构。 using Castle.Core.Resource; using Cast
我想知道我在这里的想法是否正确,我主要针对接口(interface)进行编程,所以我想知道下面的类是否应该通过 DI 注入(inject),或者我应该自己实例化一个类... 注意:这些服务保存在我的核
我是一名优秀的程序员,十分优秀!