- android - RelativeLayout 背景可绘制重叠内容
- android - 如何链接 cpufeatures lib 以获取 native android 库?
- java - OnItemClickListener 不起作用,但 OnLongItemClickListener 在自定义 ListView 中起作用
- java - Android 文件转字符串
我正在阅读直接初始化和复制初始化之间的区别 (§8.5/12):
T x(a); //direct-initialization
T y = a; //copy-initialization
我从阅读 copy-initialization 中了解到的内容是它需要 accessible & non-explicit copy-constructor ,否则程序无法编译。我通过编写以下代码验证了它:
struct A
{
int i;
A(int i) : i(i) { std::cout << " A(int i)" << std::endl; }
private:
A(const A &a) { std::cout << " A(const A &)" << std::endl; }
};
int main() {
A a = 10; //error - copy-ctor is private!
}
GCC 给出一个错误(ideone)说:
prog.cpp:8: error: ‘A::A(const A&)’ is private
到目前为止一切都很好,reaffirming what Herb Sutter says ,
Copy initialization means the object is initialized using the copy constructor, after first calling a user-defined conversion if necessary, and is equivalent to the form "T t = u;":
之后,我通过注释 private
关键字使复制构造器可访问。现在,我自然希望打印以下内容:
A(const A&)
但令我惊讶的是,它打印了这个 ( ideone ):
A(int i)
为什么?
好的,我知道首先使用
,根据需要在此处应用转换规则 (§8.5/14),然后它应该调用 copy-ctor 来初始化 10
创建了一个 A
类型的临时对象,它是 int
类型>A(int i)a
。但它没有。为什么?
如果允许实现消除调用复制构造函数的需要 (§8.5/14),那么为什么在声明复制构造函数 private
时它不接受代码?毕竟,它不是在调用它。这就像一个被宠坏的 child ,他首先生气地要一个特定玩具,当你给他一个,特定的玩具时,他会在你背后把它扔掉。 :|
这种行为会不会很危险?我的意思是,我可能会在复制构造器中做一些其他有用的事情,但如果它不调用它,那么它不会改变程序的行为吗?
最佳答案
你是在问编译器为什么要进行访问检查? C++03 中的 12.8/14:
A program is ill-formed if the copy constructor or the copy assignment operator for an object is implicitly used and the special member function is not accessible
当实现“省略复制构造”(12.8/15 允许)时,我不认为这意味着复制构造函数不再“隐式使用”,它只是没有被执行。
或者您是在问为什么标准会这样说?如果复制省略是这个关于访问检查的规则的一个异常(exception),那么您的程序在成功执行省略的实现中将是良构的,但在没有成功执行省略的实现中将是良构的。
我很确定作者会认为这是一件坏事。当然,以这种方式编写可移植代码更容易——如果您编写的代码试图复制不可复制的对象,即使在您的实现中恰好省略了复制,编译器也会告诉您。我怀疑在检查访问权限之前确定优化是否成功(或将访问权限检查推迟到尝试优化之后)也会给实现者带来不便,尽管我不知道这是否有必要考虑。
Could this behavior be dangerous? I mean, I might do some other useful thing in the copy-ctor, but if it doesn't call it, then does it not alter the behavior of the program?
当然这可能是危险的——当且仅当对象被实际复制时,复制构造函数中的副作用才会发生,你应该相应地设计它们:标准说复制可以被省略,所以不要把代码放在复制构造函数,除非您愿意在 12.8/15 中定义的条件下将其省略:
MyObject(const MyObject &other) {
std::cout << "copy " << (void*)(&other) << " to " << (void*)this << "\n"; // OK
std::cout << "object returned from function\n"; // dangerous: if the copy is
// elided then an object will be returned but you won't see the message.
}
关于c++ - 复制初始化的奇怪行为,不调用复制构造函数!,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/6163040/
C语言sscanf()函数:从字符串中读取指定格式的数据 头文件: ?
最近,我有一个关于工作预评估的问题,即使查询了每个功能的工作原理,我也不知道如何解决。这是一个伪代码。 下面是一个名为foo()的函数,该函数将被传递一个值并返回一个值。如果将以下值传递给foo函数,
CStr 函数 返回表达式,该表达式已被转换为 String 子类型的 Variant。 CStr(expression) expression 参数是任意有效的表达式。 说明 通常,可以
CSng 函数 返回表达式,该表达式已被转换为 Single 子类型的 Variant。 CSng(expression) expression 参数是任意有效的表达式。 说明 通常,可
CreateObject 函数 创建并返回对 Automation 对象的引用。 CreateObject(servername.typename [, location]) 参数 serv
Cos 函数 返回某个角的余弦值。 Cos(number) number 参数可以是任何将某个角表示为弧度的有效数值表达式。 说明 Cos 函数取某个角并返回直角三角形两边的比值。此比值是
CLng 函数 返回表达式,此表达式已被转换为 Long 子类型的 Variant。 CLng(expression) expression 参数是任意有效的表达式。 说明 通常,您可以使
CInt 函数 返回表达式,此表达式已被转换为 Integer 子类型的 Variant。 CInt(expression) expression 参数是任意有效的表达式。 说明 通常,可
Chr 函数 返回与指定的 ANSI 字符代码相对应的字符。 Chr(charcode) charcode 参数是可以标识字符的数字。 说明 从 0 到 31 的数字表示标准的不可打印的
CDbl 函数 返回表达式,此表达式已被转换为 Double 子类型的 Variant。 CDbl(expression) expression 参数是任意有效的表达式。 说明 通常,您可
CDate 函数 返回表达式,此表达式已被转换为 Date 子类型的 Variant。 CDate(date) date 参数是任意有效的日期表达式。 说明 IsDate 函数用于判断 d
CCur 函数 返回表达式,此表达式已被转换为 Currency 子类型的 Variant。 CCur(expression) expression 参数是任意有效的表达式。 说明 通常,
CByte 函数 返回表达式,此表达式已被转换为 Byte 子类型的 Variant。 CByte(expression) expression 参数是任意有效的表达式。 说明 通常,可以
CBool 函数 返回表达式,此表达式已转换为 Boolean 子类型的 Variant。 CBool(expression) expression 是任意有效的表达式。 说明 如果 ex
Atn 函数 返回数值的反正切值。 Atn(number) number 参数可以是任意有效的数值表达式。 说明 Atn 函数计算直角三角形两个边的比值 (number) 并返回对应角的弧
Asc 函数 返回与字符串的第一个字母对应的 ANSI 字符代码。 Asc(string) string 参数是任意有效的字符串表达式。如果 string 参数未包含字符,则将发生运行时错误。
Array 函数 返回包含数组的 Variant。 Array(arglist) arglist 参数是赋给包含在 Variant 中的数组元素的值的列表(用逗号分隔)。如果没有指定此参数,则
Abs 函数 返回数字的绝对值。 Abs(number) number 参数可以是任意有效的数值表达式。如果 number 包含 Null,则返回 Null;如果是未初始化变量,则返回 0。
FormatPercent 函数 返回表达式,此表达式已被格式化为尾随有 % 符号的百分比(乘以 100 )。 FormatPercent(expression[,NumDigitsAfterD
FormatNumber 函数 返回表达式,此表达式已被格式化为数值。 FormatNumber( expression [,NumDigitsAfterDecimal [,Inc
我是一名优秀的程序员,十分优秀!