- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我最近注意到一些我自己无法弄清楚的运算符重载行为。以下两个类仅在 const
上有所不同关于 ClassA
的成员比较运算符重载.在 ClassB
它们不是常量。一般来说,我知道人们总是更喜欢 const
一,但我仍然对为什么我们会看到我将在下面描述的行为感兴趣。
#include <string>
class ClassA {
public:
explicit ClassA(double t) : _t(t) {}
std::string operator<=(int const& other) const {
return "A(<=)";
}
std::string operator==(int const& other) const {
return "A(==)";
}
friend std::string operator<=(int const& other, ClassA const& expr) {
return "A'(<=)";
}
friend std::string operator==(int const& other, ClassA const& expr) {
return "A'(==)";
}
private:
double _t;
};
class ClassB {
public:
explicit ClassB(double t) : _t(t) {}
std::string operator<=(int const& other) {
return "B(<=)";
}
std::string operator==(int const& other) {
return "B(==)";
}
friend std::string operator<=(int const& other, ClassB const& expr) {
return "B'(<=)";
}
friend std::string operator==(int const& other, ClassB const& expr) {
return "B'(==)";
}
private:
double _t;
};
现在我想在
const
中使用这些类和比较函数和非常量场景。
int
main(int argc,
char* argv[]) {
ClassA a1{0};
1==a1; //OK
1<=a1; //OK
ClassA const a2{0};
1==a2; //OK
1<=a2; //OK
ClassB b1{0};
1==b1; //NOT OK
1<=b1; //OK
ClassB const b2{0};
1==b2; //OK
1<=b2; //OK
return 0;
}
一切正常,但我标记的一行
NOT OK
.这会引发编译器错误。
error C2446: '==': no conversion from 'ClassB' to 'int'
我的问题分为三个部分,但我希望有一个很好的理由可以回答所有问题。所以我希望将其发布到一个 SO 问题中仍然可以。
Why is the equality operator
==
not compiling, when the inequality<=
is? Why does it matter for the friend functions whether the member functions areconst
or not? And why does making theClassB
objectconst
fix it?
std::string operator==(int const& other)
(重新安排后)更好的匹配1==b1
.经过一番挖掘,我发现规则说这些应该在 overload resolution 的规则中生成。 .
- rewritten candidates:
- For the four relational operator expressions x<y, x<=y, x>y, and x>=y, all member, non-member, and built-in operator<=>'s found areadded to the set.
- For the four relational operator expressions x<y, x<=y, x>y, and x>=y as well as the three-way comparison expression x<=>y, asynthesized candidate with the order of the two parameters reversed isadded for each member, non-member, and built-in operator<=>'s found.
- For x!=y, all member, non-member, and built-in operator=='s found are added to the set.
- For equality operator expressions x==y and x!=y, a synthesized candidate with the order of the two parameters reversed is added foreach member, non-member, and built-in operator=='s found.
In all cases, rewritten candidates are not considered in the contextof the rewritten expression. For all other operators, the rewrittencandidate set is empty.
clang
和 gcc
带旗-std=c++2a
最新版本 x86-64 clang 12.0.0
和 x86-64 gcc 11.1
不编译,而旧版本 x86-64 clang 9.0.1
和 x86-64 gcc 9.4
做。对于 VisualStudio,我们看到了类似的模式,带有标志 /std:c++latest
.这里最新版本x64 msvc v19.28 (VS16.9)
不编译,这里直接前辈x64 msvc v19.28
做。这些测试是使用编译器资源管理器进行的 godbolt.org .clang
的编译器错误和 gcc
提示问题是std::string operator==(int const& other)
没有返回 bool
.error: return type 'std::string' (aka 'basic_string<char>') of selected 'operator==' function for rewritten '==' comparison is not 'bool'
1==b1; //NOT OK
gcc
error: return type of 'std::string ClassB::operator==(const int&)' is not 'bool'
1==b1; //NOT OK
虽然这些都是非常有趣的见解,但最初的问题仍然悬而未决。
最佳答案
没有具体的答案。但是,让我们看看 documentation .
有对返回类型没有限制 对于重载运算符(因为返回类型不参与重载解析),但有规范的实现:
...the language puts no other constraintson what the overloaded operators do, or on the return type, but in general, overloadedoperators are expected to behave as similar as possible to thebuilt-in operators
..the return types are limited by the expressions in which theoperator is expected to be used.
For example, assignment operatorsreturn by reference to make it possible to write a = b = c = d,because the built-in operators allow that.
...where built-in operators return bool, most user-defined overloads alsoreturn bool so that the user-defined operators can be used in the samemanner as the built-ins. However, in a user-defined operator overload,any type can be used as return type (including void).
If both operands have arithmetic types, or if one operand has unscopedenumeration type and the other has integral type, the usual arithmeticconversions are applied to the operands.
std::cout << (1==b1) << std::endl; // Prints B'(==)
微小的重新更新
关于C++ 比较运算符重载常量与非常量行为,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/67938508/
我想使用 NetworkX Graph 对象作为 Python dict 中的键。但是,我不希望默认的比较行为(即通过对象的地址)。相反,我希望同构图是 dict 中相同元素的键。 此行为是否已在某处
这个问题已经有答案了: What is the most effective way for float and double comparison? (33 个回答) 已关闭 7 年前。 在您认为我
我正在学习 C 编程,为了练习,我找到了一个需要解决的任务。这有点像一个游戏,有人选择一个单词,其他人猜测字母。我必须检查有多少给定的单词可能是所选单词的正确答案。 输入: 3 3//数字 n 和 m
我两天前开始学习C,在做作业时遇到了问题。我们的目的是从字符数组中获取字符列表,并通过计算连续字符并将其替换为数字来缩短它。对“a4b5c5”说“aaaabbbbbccccc”。这是我到目前为止的代码
已关闭。此问题需要 debugging details 。目前不接受答案。 编辑问题以包含 desired behavior, a specific problem or error, and the
为什么我在 if 中的比较不起作用?答案应该是 8 但它返回 0。 function findMissing(missingArray){ var getArray = missing
我想知道为什么以下 JavaScript 比较会给出不同的结果。 (1==true==1) true (2==true==2) false (0==false==0) false (0==false)
我想知道是否有人可以帮助我完成这个程序。编写一个接受两个字符串的函数。该函数应该将这两个字符串与字典顺序上排在第一位的字符串组合起来。两个字符串之间应该有一个空格。在一行上打印结果字符串。在一行上打印
有谁知道一个免费的开源库(实用程序类),它允许您比较一个 Java bean 的两个实例并返回一个属性列表/数组,这两个实例的值不同?请发布一个小样本。 干杯 托马斯 最佳答案 BeanCompara
我是java新手。任何人都可以给出以下类声明的含义 public class ListNode, V> { K key; V value; ListNode next;
我需要用 C 语言计算和比较 3 种不同大小(100 * 100、1000 * 1000 和 10000 * 10000)的 2 个矩阵相乘的执行时间。我编写了以下简单代码来为 1000 * 1000
当我在 ACCESS 2007 中运行以下 SQL 时 Select Location, COUNT(ApartmentBuildings) AS TotalIBuildingsManaged Fro
根据我对互斥锁的了解——它们通常提供对共享资源的锁定功能。因此,如果一个新线程想要访问这个锁定的共享资源——它要么退出,要么必须不断轮询锁(并在等待锁时浪费处理器周期)。 但是,监视器具有条件变量,它
通常在编程中,不应该比较浮点数据类型是否相等,因为存储的值通常是近似值。 由于两个非整数 Oracle NUMBER 值的存储方式不同(以 10 为基数),是否可以可靠地比较它们是否相等? 最佳答案
使用 PowerShell 时,我们偶尔会比较不同类型的对象。一个常见的场景是 $int -eq $bool (即其中 0 -eq $false 、 0 -ne $true 和任何非零值仅等于真,但不
#include #define MAX 1000 void any(char s1[], char s2[], char s3[]); int main() { char string1[
我想比较两个日期。 从这两个日期中,我只使用 ToShortDateString() 获取日期组件, 如下所示。现在的问题是当我比较两个日期时。它的 throw 错误—— "Operator >= c
用户输入一个数字( float 或整数),并且它必须大于下限。 这是从 UITextField 获取数字的代码: NSNumberFormatter * f = [[NSNumberFormatter
我已经摆弄这段代码大约一个小时了,它让我难以置信。我认为解决方案相当简单,但我似乎无法弄清楚。无论如何,这里去。我制作了一个 javascript 函数来检查用户输入的字符,以便它只能接受 7 个字符
我不太明白为什么当我们在不覆盖 equals 的情况下比较具有相同类属性的两个实例时方法,它将给出 false .但它会给出 true当我们比较一个案例类的两个实例时。例如 class A(val
我是一名优秀的程序员,十分优秀!