- mongodb - 在 MongoDB mapreduce 中,如何展平值对象?
- javascript - 对象传播与 Object.assign
- html - 输入类型 ="submit"Vs 按钮标签它们可以互换吗?
- sql - 使用 MongoDB 而不是 MS SQL Server 的优缺点
我最近从 Scott Meyers 那里购买了新的 Effective现代 C++ 并通读了它。但是我遇到了一件让我非常烦恼的事情。
在第 5 项中,Scott 说使用 auto
是一件很棒的事情。它可以节省输入,在大多数情况下为您提供正确的类型,并且可能不受类型不匹配的影响。我完全理解这一点并想到了 auto
也是一件好事。
但是在第 6 项中,斯科特说每个硬币都有两个面。同样,可能存在 auto
的情况推导出完全错误的类型,例如用于代理对象。
你可能已经知道这个例子:
class Widget;
std::vector<bool> features(Widget w);
Widget w;
bool priority = features(w)[5]; // this is fine
auto priority = features(w)[5]; // this result in priority being a proxy
// to a temporary object, which will result
// in undefined behavior on usage after that
// line
auto priority = static_cast<bool>(features(w)[5]);
auto
的两个优势在显式给定类型上。
auto priority = static_cast<bool>(features(w)[5]);
bool priority = features(w)[5];
Guideline: Consider declaring local variables auto x = type{ expr }; when you do want to explicitly commit to a type. It is self-documenting to show that the code is explicitly requesting a conversion, it guarantees the variable will be initialized, and it won’t allow an accidental implicit narrowing conversion. Only when you do want explicit narrowing, use ( ) instead of { }.
bool priority = features(w)[5];
auto priority = static_cast<bool>(features(w)[5]);
auto priority = bool(features(w)[5]);
auto priority = bool{features(w)[5]};
最佳答案
遵循 C++ 标准:
§ 8.5 Initializers
[dcl.init]
The initialization that occurs in the form
T x = a;
as well as in argument passing, function return, throwing an exception (15.1), handling an exception (15.3), and aggregate member initialization (8.5.1) is called copy-initialization.
auto x = features(w)[5];
template <typename A>
void foo(A x) {}
foo(features(w)[5]);
auto bar()
{
return features(w)[5];
}
auto lambda = [] (auto x) {};
lambda(features(w)[5]);
static_cast<T>
移动到赋值的左侧”。
/*1*/ foo(static_cast<bool>(features(w)[5]));
/*2*/ return static_cast<bool>(features(w)[5]);
/*3*/ lambda(static_cast<bool>(features(w)[5]));
static_cast<T>
是一种强制所需类型的优雅方式,或者可以通过显式构造函数调用来表示:
foo(bool{features(w)[5]});
Whenever you want to force the type of a variable, use
auto x = static_cast<T>(y);
instead ofT x{y};
.
The type inference with
auto
is cool, but may end up with undefined behavior if used unwisely.
If the compiler's regular type-deduction mechanism is not what you want, use
static_cast<T>(y)
.
bool priority = features(w)[5];
auto priority = static_cast<bool>(features(w)[5]);
auto priority = bool(features(w)[5]);
auto priority = bool{features(w)[5]};
std::vector<bool>::reference
是
不是隐含的 可转换为
bool
:
struct BoolReference
{
explicit operator bool() { /*...*/ }
};
bool priority = features(w)[5];
将
不编译 ,因为它不是一个明确的 bool 上下文。其他的会正常工作(只要
operator bool()
是可访问的)。
std::vector<bool>::reference
以旧方式实现,尽管转换运算符不是
explicit
,它返回
int
反而:
struct BoolReference
{
operator int() { /*...*/ }
};
auto priority = bool{features(w)[5]};
初始化,如使用
{}
防止缩小(将
int
转换为
bool
是)。
bool
呢?完全没有,但关于一些用户定义的类型,令我们惊讶的是,声明
explicit
构造函数:
struct MyBool
{
explicit MyBool(bool b) {}
};
MyBool priority = features(w)[5];
初始化将
不编译 ,因为复制初始化语法需要非显式构造函数。其他人会工作。
auto priority = bool{features(w)[5]};
features(w)[5]
真的是 .
auto x = T{y};
的显式类型初始化器的观点。形式(尽管它与
auto x = static_cast<T>(y)
不同,因此并非所有参数都适用)超过
T x{y};
, 哪个是:
auto
变量必须始终被初始化。也就是说,你不能写auto a;
,就像你可以写容易出错的 int a;
auto f = 3.14f;
// ^ float
auto s = "foo"s;
// ^ std::string
auto func(double) -> int;
auto func = [=] (double) {};
using dict = set<string>;
template <class T>
using myvec = vector<T, myalloc>;
auto x = T{y};
<category> name = <type> <initializer>;
T x{y}
相比,使用复制省略和非显式复制/移动构造函数,它的成本为零。句法。 unique_ptr<Base> p = make_unique<Derived>(); // subtle difference
auto p = unique_ptr<Base>{make_unique<Derived>()}; // explicit and clear
{}
保证没有隐式转换和缩小。 auto x = T{}
的一些缺点。一般形式,在这篇文章中已经描述过:
auto x = std::atomic<int>{}; // fails to compile, copy constructor deleted
-fno-elide-constructors
),则移动不可移动类型会导致昂贵的复制: auto a = std::array<int,50>{};
关于c++ - 为什么我更喜欢 "explicitly typed initializer"成语而不是明确给出类型,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/25607216/
假设a是张量,那么有什么区别: 类型(a) a.类型 a.type() 我找不到区分这些的文档。 最佳答案 type 是 python 内置方法。 它将返回对象的类型。喜欢 torch.Tensor.
什么是 Type 1 的居民的例子?两者都不是 Type也不是Type的居民?在 Idris REPL 中进行探索时,我无法想出任何东西。 更准确地说,我正在寻找一些 x除了 Type产生以下结果:
我找到了一些资源,但我不确定我是否理解。 我找到的一些资源是: http://help.sap.com/saphelp_nw70/helpdata/en/fc/eb2ff3358411d1829f00
这两个函数原型(prototype)有什么区别? void apply1(double(f)(double)); void apply2(double(*f)(double)); 如果目标是将提供的函
http://play.golang.org/p/icQO_bAZNE 我正在练习使用堆进行排序,但是 prog.go:85: type bucket is not an expression
假设有一个泛型定义的方法信息对象,即一个方法信息对象,这样的方法Info.IsGenericMethodDefinition==TRUE:。也可以说它们也有一个泛型参数列表:。我可以使用以下命令获取该
在具有依赖类型的语言中,您可以使用 Type-in-Type 来简化语言并赋予它很多功能。这使得语言在逻辑上不一致,但如果您只对编程感兴趣而不对定理证明感兴趣,这可能不是问题。 在 Cayenne
根据 Nim 手册,变量类型是“静态类型”,而变量在内存中指向的实际值是“动态类型”。 它们怎么可能是不同的类型?我认为将错误的类型分配给变量将是一个错误。 最佳答案 import typetrait
假设您有以下结构和协议(protocol): struct Ticket { var items: [TicketItem] = [] } struct TicketItem { } prot
我正在处理一个 EF 问题,我发现它很难调试...以前,在我的系统中有一个表类型继承设置管理不同的用户类型 - 所有用户共有的一种根类型,以及大致基于使用该帐户的人员类型的几种不同的子类型。现在,我遇
这是我的 DBManager.swift import RealmSwift class DBManager { class func getAllDogs() -> [Dog] {
我正在尝试使用傅里叶校正图像中的曝光。这是我面临的错误 5 padded = np.log(padded + 1) #so we never have log of 0 6 g
关闭。这个问题是opinion-based .它目前不接受答案。 想要改进这个问题? 更新问题,以便 editing this post 可以用事实和引用来回答它. 关闭 9 年前。 Improve
请考虑以下设置: protocol MyProcotol { } class MyModel: MyProcotol { } enum Result { case success(value:
好吧,我将我的 python 项目编译成一个可执行文件,它在我的电脑上运行,但我将它发送给几个 friend 进行测试,他们都遇到了这个错误。我以前从未见过这样的错误。我使用 Nuitka 来编译代码
当我尝试训练我的模型时"ValueError: Type must be a sub-type of ndarray type"出现在 line x_norm=(np.power(x,2)).sum(
我尝试在另一个类中打断、计数然后加入对象。所以我构建协议(protocol): typealias DataBreaker = () -> [Double] typealias DataJoiner
我正在使用 VS 2015 更新 3、Angular 2.1.2、Typescript 2.0.6 有人可以澄清什么是 typings 与 npm @types 以及本月很难找到的任何其他文档吗? 或
我正在考虑从 VS2010 更改为 Mono,因此我通过 MoMA 运行我的程序集,看看我在转换过程中可能遇到多少困难。在生成的报告中,我发现我不断收到此错误: bool Type.op_Equali
主要问题 不太确定这是否可能,但由于我讨厌 Typescript 并且它使我的编码变得困难,我想我会问只是为了确定。 interface ISomeInterface { handler: ()
我是一名优秀的程序员,十分优秀!