- mongodb - 在 MongoDB mapreduce 中,如何展平值对象?
- javascript - 对象传播与 Object.assign
- html - 输入类型 ="submit"Vs 按钮标签它们可以互换吗?
- sql - 使用 MongoDB 而不是 MS SQL Server 的优缺点
我一直在使用解析为与声明相同类型的定义中的推导返回类型。这有效:
template <typename>
struct Cls {
static std::size_t f();
};
template <typename T>
decltype(sizeof(int)) Cls<T>::f() { return 0; }
sizeof(int)
将定义更改为应该等效的内容与
sizeof(T)
it fails
template <typename T>
decltype(sizeof(T)) Cls<T>::f() { return 0; }
error: prototype for ‘decltype (sizeof (T)) Cls<T>::f()’ does not match any in class ‘Cls<T>’
decltype(sizeof(T)) Cls<T>::f() { return 0; }
^~~~~~
so.cpp:4:24: error: candidate is: static std::size_t Cls<T>::f()
static std::size_t f();
^
template <typename>
struct Cls {
static void f(std::size_t);
};
template <typename T>
void Cls<T>::f(decltype(sizeof(T))) { } // sizeof(int) works instead
decltype(sizeof(T))
它编译成功,我可以
static_assert
返回类型是
size_t
.以下编译成功:
#include <type_traits>
template <typename T>
struct Cls {
static decltype(sizeof(T)) f();
};
template <typename T>
decltype(sizeof(T)) Cls<T>::f() { return 0; }
static_assert(std::is_same<std::size_t, decltype(Cls<int>::f())>{}, "");
template <int I>
struct Cls {
static int f();
};
template <int I>
decltype(I) Cls<I>::f() { return I; }
decltype(I)
如果我使用
int
在定义和声明中都有效在定义和声明中它都有效,但两者不同则失败。
Cls
改成不是类模板,编译成功。
template <typename>
struct Cls {
static int f();
using Integer = decltype(Cls::f());
};
template <typename T>
typename Cls<T>::Integer Cls<T>::f() { return I; }
struct S {
template <int N>
int f();
};
template <int N>
decltype(N) S::f() {}
template <int I>
那样依赖它也会受到影响以上?
最佳答案
因为当涉及到模板参数时,decltype
根据标准返回一个独特的依赖类型,见下文。如果没有模板参数,那么它解析为一个明显的 size_t
.因此,在这种情况下,您必须选择声明和定义都具有独立表达式(例如 size_t/decltype(sizeof(int))
)作为返回类型,或者两者都具有依赖表达式(例如 decltype(sizeof(T))
),它们解析为唯一的依赖类型并考虑是等价的,如果它们的表达式是等价的(见下文)。
在这篇文章中,我使用了 C++ 标准草案 N3337 .
§ 7.1.6.2 [dcl.type.simpl]
¶ 4
The type denoted by decltype(e) is defined as follows: — if e is an unparenthesized id-expression or an unparenthesized class member access (5.2.5), decltype(e) is the type of the entity named by e. If there is no such entity, or if e names a set of overloaded func- tions, the program is ill-formed;
— otherwise, if e is an xvalue, decltype(e) is T&&, where T is the type of e;
— otherwise, if e is an lvalue, decltype(e) is T&, where T is the type of e;
— otherwise, decltype(e) is the type of e.
decltype(sizeof(int))
.但是对于
decltype(sizeof(T))
还有另一部分解释它是什么。
§ 14.4 [temp.type]
¶ 2
If an expression e involves a template parameter, decltype(e) denotes a unique dependent type. Two such decltype-specifiers refer to the same type only if their expressions are equivalent (14.5.6.1). [ Note: however, it may be aliased, e.g., by a typedef-name. — end note ]
lib/AST/Type.cpp
中的 Clang LLVM 源版本 3.9 中
DecltypeType::DecltypeType(Expr *E, QualType underlyingType, QualType can)
// C++11 [temp.type]p2: "If an expression e involves a template parameter,
// decltype(e) denotes a unique dependent type." Hence a decltype type is
// type-dependent even if its expression is only instantiation-dependent.
: Type(Decltype, can, E->isInstantiationDependent(),
E->isInstantiationDependent(),
E->getType()->isVariablyModifiedType(),
E->containsUnexpandedParameterPack()),
lib/AST/ASTContext.cpp
中的 Clang 源 3.9 版中
QualType ASTContext::getDecltypeType(Expr *e, QualType UnderlyingType) const {
DecltypeType *dt;
// C++11 [temp.type]p2:
// If an expression e involves a template parameter, decltype(e) denotes a
// unique dependent type. Two such decltype-specifiers refer to the same
// type only if their expressions are equivalent (14.5.6.1).
if (e->isInstantiationDependent()) {
llvm::FoldingSetNodeID ID;
DependentDecltypeType::Profile(ID, *this, e);
void *InsertPos = nullptr;
DependentDecltypeType *Canon
= DependentDecltypeTypes.FindNodeOrInsertPos(ID, InsertPos);
if (!Canon) {
// Build a new, canonical typeof(expr) type.
Canon = new (*this, TypeAlignment) DependentDecltypeType(*this, e);
DependentDecltypeTypes.InsertNode(Canon, InsertPos);
}
dt = new (*this, TypeAlignment)
DecltypeType(e, UnderlyingType, QualType((DecltypeType *)Canon, 0));
} else {
dt = new (*this, TypeAlignment)
DecltypeType(e, UnderlyingType, getCanonicalType(UnderlyingType));
}
Types.push_back(dt);
return QualType(dt, 0);
}
decltype
的那些独特的依赖类型。在/来自一个特殊的集合。
decltype
的表达式是
sizeof(T)
那总是
size_t
?是的,这对人类读者来说是显而易见的。但是当你设计和实现一个正式的语法和语义规则时,尤其是对于像 C++ 这样复杂的语言,你必须把问题分组并为它们定义规则,而不是仅仅为每个特定的问题想出一个规则,在后者这样你就无法随着你的语言/编译器设计而移动。这里同样没有公正的规则:if
decltype
有一个不需要任何模板参数解析的函数调用表达式 - 解析
decltype
到函数的返回类型。不仅如此,您还需要涵盖很多情况,因此您提出了一个更通用的规则,就像上面引用的标准(
14.4[2]
)一样。
auto
类似的非明显案例。 ,
decltype(auto)
由 AndyG 在 C++-14 中发现(N4296,§ 7.1.6.4 [dcl.spec.auto] 12/13):
§ 7.1.6.4 [dcl.spec.auto]
¶ 13
Redeclarations or specializations of a function or function template with a declared return type that uses a placeholder type shall also use that placeholder, not a deduced type. [ Example:
auto f();
auto f() { return 42; } // return type is int
auto f(); // OK
int f(); // error, cannot be overloaded with auto f()
decltype(auto) f(); // error, auto and decltype(auto) don’t match
§ 17.4 (old § 14.4) [temp.type]
¶ 2
If an expression e is type-dependent (17.6.2.2), decltype(e) denotes a unique dependent type. Two such decltype-specifiers refer to the same type only if their expressions are equivalent (17.5.6.1). [ Note: however, such a type may be aliased, e.g., by a typedef-name. — end note ]
§ 17.6.2.2 [temp.dep.expr] (old § 14.6.2.2)
¶ 4
Expressions of the following forms are never type-dependent (because the type of the expression cannot be dependent):
...
sizeof ( type-id )
...
sizeof
如果
type-id
可以依赖于值依赖。值依赖表达式与
decltype
之间没有关系.想了想,没找到原因
decltype(sizeof(T))
不得或不能解析为
size_t
.而且我认为这是编译器开发人员不太注意的标准中非常偷偷摸摸的更改(“涉及模板参数”到“类型相关”)(可能被许多其他更改所淹没,也许不认为它可能实际上改变一些东西,只是一个简单的配方改进)。更改确实有意义,因为
sizeof(T)
不依赖于类型,而是依赖于值。
decltype(e)
的操作数是一个未计算的操作数,即不关心值,只关心类型。这就是为什么
decltype
仅当
e
时才返回唯一类型是类型相关的。
sizeof(e)
可能只是值(value)依赖。
-std=c++1z
尝试了代码- 相同的结果:错误。我更进一步并尝试了以下代码:
template <typename>
struct Cls {
static std::size_t f();
};
template <typename T>
decltype(sizeof(sizeof(T))) Cls<T>::f() {
return 0;
}
sizeof(sizeof(T))
既不依赖于类型,也不依赖于值(见
this post)。这让我有理由假设编译器正在以 C++-11/14 标准的旧方式工作(即“涉及模板参数”),就像上面来自 clang 3.9 源代码的源代码片段(我可以验证最新开发的clang 5.0 具有相同的行,在标准中没有发现与此新更改相关的任何内容),但不依赖于类型。
关于c++ - 在没有 decltype 的情况下声明时,函数定义中 decltype 中的依赖类型或参数无法编译,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/44294743/
我在覆盖 ReSwift Pod 中的函数时遇到问题。我有以下模拟类(class): import Foundation import Quick import Nimble import RxSwi
我有一个类似于下面的继承结构。我正在采用 Printable 协议(protocol)并努力覆盖 description 属性。我遇到了一个谷歌此时似乎不知道的奇怪错误,提示为第三类,并引用了第二类和
我有一个类“Cat”和 Cat 类的一个子类“DerivedCat”。 Cat 有一个函数 meow(),而 DerivedCat 覆盖了这个函数。 在应用程序中,我声明了一个 Cat 对象: Cat
Kotlin 变量 变量是用于存储数据值的容器。 要创建一个变量,使用 var 或 val,然后使用等号(=)给它赋值: 语法 var 变量名 = 值 val 变量名 = 值 示例 va
C 中的所有标识符在使用前都需要声明,但我找不到它在 C99 标准中表示的位置。 我觉得也是指宏定义,不过定义的只是宏展开顺序。 最佳答案 C99:TC3 6.5.1 §2,脚注 79 明确指出: T
今天我的博客提要显示错误: This page contains the following errors: error on line 2 at column 6: XML declaration
在编写 IIF 语句、表和下面给出的语句时出现错误。 陈述: SELECT IIF(EMP_ID=1,'True','False') from Employee; table : CREATE TAB
我正在创建一个登录 Activity ,我希望它在按下登录按钮时显示进度对话框,我声明、初始化并调用了它,但它没有显示。但是当我在创建时调用进度对话框时,它出现了 这是我的代码: public cla
当我输入声明语句时: Vector distance_vector = new Vector(); 我收到错误(在两种情况下都在“双”下划线): Syntax error on token "doub
我正在本地部署在docker-for-desktop中。这样我将来可以迁移到kubernetes集群。 但是我面临一个问题。使用永久卷时,docker容器/ pod中的目录将被覆盖。 我正在拉最新的S
我有一个 MyObject 类型的对象 obj,我声明了它的实例。 MyObject obj; 但是,我没有初始化它。 MyObject 的类看起来像: public class MyObject {
关闭。这个问题是opinion-based 。目前不接受答案。 想要改进这个问题吗?更新问题,以便 editing this post 可以用事实和引文来回答它。 . 已关闭 9 年前。 Improv
这个问题已经有答案了: Android: Issue during Arraylist declaration (1 个回答) 已关闭 9 年前。 有时我会看到 ArrayList 声明如下 Arra
我对java比较陌生,经过大量搜索,我无法将相关问题的任何解决方案与我的解决方案配对。我正在尝试实现一种非常简单的方法来写入/读取数组,但编译器无法识别它。 “键盘”也是一个“无法识别的变量”。这是数
简短:何时分配内存 - 在声明或初始化时? 长整型:int x;将占用与int z = 10;相同的内存。 此外,这对于包含更多数据的自定义对象将如何工作。假设我有这个对象: public class
我需要使用此程序更好地理解函数定义、声明和正确调用。我真的需要了解如何使用它们。您能否向我展示编写此程序的正确方法(所有三个都正确并进行解释)? #include #include quad_eq
这是我的主要功能以及我要传递的内容。 int main(void){ struct can elC[7]; // Create an array of stucts Initiali
我想知道是否有更好的方法来完成此任务; 我有一个对象 - 其中一个属性是字典。我有一组逗号分隔值。我需要过滤 Dictionary 并仅获取 Dictionary 值至少与其中一个值匹配的那些元素 这
下面的using-declarations有什么意义 using eoPop::size; using eoPop::operator[]; using eoPop::back; using eoPo
我的问题更像是一个关于 for 循环样式的好奇问题。在阅读别人的一些旧代码时,我遇到了一种我以前从未见过的风格。 var declaredEarlier = Array for(var i=0, le
我是一名优秀的程序员,十分优秀!