- Java 双重比较
- java - 比较器与 Apache BeanComparator
- Objective-C 完成 block 导致额外的方法调用?
- database - RESTful URI 是否应该公开数据库主键?
我正在阅读 Scott Meyers 的《Effective C++: 55 Specific Ways to Improve Your Programs and Designs》,他说:
Having a function return a constant value is generally inappropriate, but sometimes doing so can reduce the incidence of client errors without giving up safety or efficiency. For example, consider the declaration of the operator* function:
class Rational { ... };
const Rational operator*(const Rational& lhs, const Rational& rhs);
根据 Meyers 的说法,这样做可以防止像这样的“暴行”,如果 a、b 是原始类型,这将是非法的:
Rational a, b, c;
...
(a * b) = c;
这让我很困惑,在试图理解为什么上述赋值对于原始类型而不是用户定义类型是非法的时,我遇到了右值和左值
在看了一些SO问题后,我仍然觉得我对右值和左值没有很好的理解,但这是我的基本理解:左值引用内存中的一个位置,因此可以分配给(它可以是在 = 运算符的两侧也是如此);然而,右值不能被分配,因为它不引用内存位置(例如,函数返回和文字的临时值)
我的问题是:为什么分配给两个数字/对象的乘积对于用户定义的类型是合法的(即使它没有意义)而不是基元?它与返回类型有关吗?重载的 * 运算符返回可分配值还是临时值?
最佳答案
[expr.call]/14:
A function call is an lvalue if the result type is an lvalue reference type or an rvalue reference to function type, an xvalue if the result type is an rvalue reference to object type, and a prvalue otherwise.
这是有道理的,因为结果没有“名字”。如果您返回一个引用,则意味着它是对某处某个对象的引用,该对象确实“有名称”(通常但并非总是如此)。
然后是这个:
[expr.ass]/1:
The assignment operator (=) and the compound assignment operators all group right-to-left. All require a modifiable lvalue as their left operand; their result is an lvalue referring to the left operand.
这是说赋值需要在左侧有一个左值。到目前为止,一切都很好;你自己已经介绍过了。
那么非 const
函数调用结果如何工作?
根据特殊规则!
[over.oper]/8:
: [..] Some predefined operators, such as +=, require an operand to be an lvalue when applied to basic types; this is not required by operator functions.
… 和应用于类类型对象的 =
调用运算符函数。
我不能轻易回答“为什么”:从表面上看,在处理类时放宽此限制是有道理的,而对内置函数的原始(继承)限制似乎总是有点过分(在我的意见),但出于兼容性原因必须保留。
但是,像 Meyers 这样的人指出,现在返回 const
值以有效地“撤消”此更改变得有用(有点)。
最终,无论哪种方式,我都不会努力寻找强有力的理由。
关于c++ - 重载运算符返回什么类型的值(对于用户自定义类型): rvalue or lvalue?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/57590241/
Or 运算符 对两个表达式进行逻辑“或”运算。 result = expression1 Or expression2 参数 result 任意数值变量。 expression1 任意
Not 运算符 对表达式执行逻辑非运算。 result = Not expression 参数 result 任意数值变量。 expression 任意表达式。 说明 下表显示如何
Is 运算符 比较两个对象引用变量。 result = object1 Is object2 参数 result 任意数值变量。 object1 任意对象名。 object2 任意
\ 运算符 两个数相除并返回以整数形式表示的结果。 result = number1\number2 参数 result 任意数值变量。 number1 任意数值表达式。 numbe
And 运算符 对两个表达式进行逻辑“与”运算。 result = expression1 And expression2 参数 result 任意数值变量。 expression1
运算符(+) 计算两个数之和。 result = expression1 + expression2 参数 result 任意数值变量。 expression1 任意表达式。 exp
我对此感到困惑snippet : var n1 = 5-"4"; var n2 = 5+"4"; alert(n1); alert(n2); 我知道 n1 是 1。那是因为减号运算符会将字符串“4”转
我想我会得到 12,而不是 7。 w++,那么w就是4,也就是100,而w++, w 将是 8,1000;所以 w++|z++ 将是 100|1000 = 1100 将是 12。 我怎么了? int
Xor 运算符 对两个表达式进行逻辑“异或”运算。 result = expression1 Xor expression2 参数 result 任意数值变量。 expression1
Mod 运算符 两个数值相除并返回其余数。 result = number1 Mod number2 参数 result 任意数值变量。 number1 任意数值表达式。 numbe
Imp 运算符 对两个表达式进行逻辑蕴涵运算。 result = expression1 Imp expression2 参数 result 任意数值变量。 expression1 任
Eqv 运算符 执行两个表达式的逻辑等价运算。 result = expression1 Eqv expression2 参数 result 任意数值变量。 expression1 任
我有一个运算符重载的简单数学 vector 类。我想为我的运算符(operator)获取一些计时结果。我可以通过计时以下代码轻松计时我的 +=、-=、*= 和/=: Vector sum; for(s
我是用户定义比较运算符的新手。我正在读一本书,其中提到了以下示例: struct P { int x, y; bool operator、运算符<等),我们
在 SQL 的维基百科页面上,有一些关于 SQL 中 bool 逻辑的真值表。 [1] 维基百科页面似乎来源于 SQL:2003 标准。 等号运算符 (=) 的真值表与 SQL:2003 草案中的 I
我遇到了一个奇怪的 C++ 运算符。 http://www.terralib.org/html/v410/classoracle_1_1occi_1_1_number.html#a0f2780081f
我正在阅读关于 SO 和 answers 中的一个问题,它被提到为: If no unambiguous matching deallocation function can be found, pr
我偶然发现了这个解决方案,但我无法理解其中到底发生了什么。谁能解释一下! 据我了解,它试图通过计算一半的单元格然后将其加倍来计算 a*b 网格中的单元格数量。但是我无法理解递归调用。 请不要建议其他解
Go的基本类型 布尔类型bool 长度:1字节 取值:布尔类型的取值只能是true或者false,不能用数字来表示 整型 通用整型 int / uint(有符号 / 无符号,下面也类似) 长度:根据运
在本教程中,您将学习JavaScript中可用的不同运算符,以及在示例的帮助下如何使用它们。 什么是运算符? 在JavaScript中,运算符是一种特殊符号,用于对运算数(值和变量)执行操作。例如,
我是一名优秀的程序员,十分优秀!