- mongodb - 在 MongoDB mapreduce 中,如何展平值对象?
- javascript - 对象传播与 Object.assign
- html - 输入类型 ="submit"Vs 按钮标签它们可以互换吗?
- sql - 使用 MongoDB 而不是 MS SQL Server 的优缺点
C++ 标准是否保证 (x!=y)
始终与 !(x==y)
具有相同的真值?
我知道这里涉及许多微妙之处:运算符 ==
和 !=
可能被重载。它们可能被重载以具有不同的返回类型(只需隐式转换为 bool
)。甚至 !
操作符也可能在返回类型上被重载。这就是为什么我在上面提到了“真值”,但试图进一步阐述它,利用到 bool
的隐式转换,并试图消除可能的歧义:
bool ne = (x!=y);
bool e = (x==y);
bool result = (ne == (!e));
这里是否保证result
为true
?
C++ 标准在 5.10 节中指定了相等运算符,但似乎主要是在句法上定义它们(以及一些关于指针比较的语义)。存在的概念 EqualityComparable存在,但没有专门说明其运算符 ==
与 !=
运算符的关系。
存在related documents from C++ working groups ,话说……
It is vital that equal/unequal [...] behave as boolean negations of each other. After all, the world would make no sense if both operator==() and operator!=() returned false! As such, it is common to implement these operators in terms of each other
但是,这仅反射(reflect)了 Common Sense™,并没有指定它们必须像这样实现。
一些背景知识:我只是想编写一个函数来检查两个值(未知类型)是否相等,如果不是这样,则打印一条错误消息。我想说这里需要的概念是类型是EqualityComparable
。但是为此,仍然需要编写 if (!(x==y)) {…}
并且可以not 编写 if (x!=y ) {...}
,因为这将使用不同的运算符,它根本没有被 EqualityComparable
的概念所涵盖,甚至可能以不同的方式重载...
我知道程序员基本上可以在他的自定义重载中为所欲为。我只是想知道他是否真的允许做所有事情,或者是否有标准强加的规则。也许这些微妙的陈述之一表明偏离通常的实现会导致未定义的行为,例如 NathanOliver mentioned in a comment, but which seemed to only refer to certain types .例如,标准明确声明对于容器类型,a!=b
等价于!(a==b)
(第 23.2.1 节,表 95,“容器要求”)。
但是对于一般的、用户定义的类型,目前似乎没有这样的要求。这个问题被标记为 language-lawyer
,因为我希望有一个明确的声明/引用,但我知道这几乎是不可能的:虽然有人可以指出它说运算符(operator) 必须相互否定,很难证明该标准的约 1500 页没有一个是这样说的......
有疑问,除非有进一步的提示,否则我稍后会支持/接受相应的答案,现在假设比较 EqualityComparable
类型的不相等性应该使用 if (!(x==y))
为了安全起见。
最佳答案
Does the C++ standard guarantee that
(x!=y)
always has the same truth value as!(x==y)
?
不,它没有。绝对没有什么能阻止我写作:
struct Broken {
bool operator==(const Broken& ) const { return true; }
bool operator!=(const Broken& ) const { return true; }
};
Broken x, y;
这是格式完美的代码。从语义上讲,它被破坏了(顾名思义),但从纯 C++ 代码功能的角度来看,它肯定没有错。
该标准在 [over.oper]/7 中也明确表明这是可以的。 :
The identities among certain predefined operators applied to basic types (for example,
++a ≡ a+=1
) need not hold for operator functions. Some predefined operators, such as+=
, require an operand to be an lvalue when applied to basic types; this is not required by operator functions.
同样,C++ 标准中没有任何内容可以保证 operator<
实际上实现了一个有效的 Ordering (或 x<y <==> !(x>=y)
等)。一些标准库实现实际上会添加工具以尝试在有序容器中为您调试它,但这只是实现质量问题,而不是基于标准的决策。
库解决方案,如 Boost.Operators存在至少可以让程序员更容易做到这一点:
struct Fixed : equality_comparable<Fixed> {
bool operator==(const Fixed&) const;
// a consistent operator!= is provided for you
};
在 C++14 中,Fixed
不再是基类的聚合。但是,在 C++17 中,它又是一个聚合(通过 P0017 的方式)。
通过P1185对于 C++20,库解决方案实际上已成为一种语言解决方案 - 您只需编写以下代码:
struct Fixed {
bool operator==(Fixed const&) const;
};
bool ne(Fixed const& x, Fixed const& y) {
return x != y;
}
ne()
的正文成为有效表达式,计算结果为 !x.operator==(y)
-- 所以你不必担心保持两个比较一致,也不必依赖库解决方案来帮助。
关于c++ - 等式运算符重载 : Is (x! =y) == (!(x==y))?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/35943551/
#include using namespace std; class C{ private: int value; public: C(){ value = 0;
这个问题已经有答案了: What is the difference between char a[] = ?string?; and char *p = ?string?;? (8 个回答) 已关闭
关闭。此题需要details or clarity 。目前不接受答案。 想要改进这个问题吗?通过 editing this post 添加详细信息并澄清问题. 已关闭 7 年前。 此帖子已于 8 个月
除了调试之外,是否有任何针对 c、c++ 或 c# 的测试工具,其工作原理类似于将独立函数复制粘贴到某个文本框,然后在其他文本框中输入参数? 最佳答案 也许您会考虑单元测试。我推荐你谷歌测试和谷歌模拟
我想在第二台显示器中移动一个窗口 (HWND)。问题是我尝试了很多方法,例如将分辨率加倍或输入负值,但它永远无法将窗口放在我的第二台显示器上。 关于如何在 C/C++/c# 中执行此操作的任何线索 最
我正在寻找 C/C++/C## 中不同类型 DES 的现有实现。我的运行平台是Windows XP/Vista/7。 我正在尝试编写一个 C# 程序,它将使用 DES 算法进行加密和解密。我需要一些实
很难说出这里要问什么。这个问题模棱两可、含糊不清、不完整、过于宽泛或夸夸其谈,无法以目前的形式得到合理的回答。如需帮助澄清此问题以便重新打开,visit the help center . 关闭 1
有没有办法强制将另一个 窗口置于顶部? 不是应用程序的窗口,而是另一个已经在系统上运行的窗口。 (Windows, C/C++/C#) 最佳答案 SetWindowPos(that_window_ha
假设您可以在 C/C++ 或 Csharp 之间做出选择,并且您打算在 Windows 和 Linux 服务器上运行同一服务器的多个实例,那么构建套接字服务器应用程序的最明智选择是什么? 最佳答案 如
你们能告诉我它们之间的区别吗? 顺便问一下,有什么叫C++库或C库的吗? 最佳答案 C++ 标准库 和 C 标准库 是 C++ 和 C 标准定义的库,提供给 C++ 和 C 程序使用。那是那些词的共同
下面的测试代码,我将输出信息放在注释中。我使用的是 gcc 4.8.5 和 Centos 7.2。 #include #include class C { public:
很难说出这里问的是什么。这个问题是含糊的、模糊的、不完整的、过于宽泛的或修辞性的,无法以目前的形式得到合理的回答。如需帮助澄清此问题以便重新打开它,visit the help center 。 已关
我的客户将使用名为 annoucement 的结构/类与客户通信。我想我会用 C++ 编写服务器。会有很多不同的类继承annoucement。我的问题是通过网络将这些类发送给客户端 我想也许我应该使用
我在 C# 中有以下函数: public Matrix ConcatDescriptors(IList> descriptors) { int cols = descriptors[0].Co
我有一个项目要编写一个函数来对某些数据执行某些操作。我可以用 C/C++ 编写代码,但我不想与雇主共享该函数的代码。相反,我只想让他有权在他自己的代码中调用该函数。是否可以?我想到了这两种方法 - 在
我使用的是编写糟糕的第 3 方 (C/C++) Api。我从托管代码(C++/CLI)中使用它。有时会出现“访问冲突错误”。这使整个应用程序崩溃。我知道我无法处理这些错误[如果指针访问非法内存位置等,
关闭。这个问题不符合Stack Overflow guidelines .它目前不接受答案。 我们不允许提问寻求书籍、工具、软件库等的推荐。您可以编辑问题,以便用事实和引用来回答。 关闭 7 年前。
已关闭。此问题不符合Stack Overflow guidelines 。目前不接受答案。 要求我们推荐或查找工具、库或最喜欢的场外资源的问题对于 Stack Overflow 来说是偏离主题的,因为
我有一些 C 代码,将使用 P/Invoke 从 C# 调用。我正在尝试为这个 C 函数定义一个 C# 等效项。 SomeData* DoSomething(); struct SomeData {
这个问题已经有答案了: Why are these constructs using pre and post-increment undefined behavior? (14 个回答) 已关闭 6
我是一名优秀的程序员,十分优秀!