为什么找不到使用命名空间中定义的类型实例化的 std::weak_ptr 的重载运算符==?
Why overloaded operator== for std::weak_ptr instantiated with type defined in namespace can't be found?
我正在使用Visual Studio 2015。
知道为什么编译此代码:
#include <memory>
class Foo;
class Bar;
typedef std::pair<Foo*,std::weak_ptr<Bar>> Object;
typedef std::vector<Object> ObjectVect;
bool operator==( std::weak_ptr<Bar> left,
std::weak_ptr<Bar> right )
{
return left.lock() == right.lock();
}
int main( int argc, char* argv[] )
{
ObjectVect vect;
Object obj;
auto foundIter = std::find( vect.begin(), vect.end(), obj );
return 0;
}
虽然这个给了我错误:
#include <memory>
class Foo;
namespace MyNamespace
{
class Bar;
}
typedef std::pair<Foo*,std::weak_ptr<MyNamespace::Bar>> Object;
typedef std::vector<Object> ObjectVect;
bool operator==( std::weak_ptr<MyNamespace::Bar> left,
std::weak_ptr<MyNamespace::Bar> right )
{
return left.lock() == right.lock();
}
int main( int argc, char* argv[] )
{
ObjectVect vect;
Object obj;
auto foundIter = std::find( vect.begin(), vect.end(), obj );
return 0;
}
错误 C2678 二进制"==":找不到需要左手的运算符 类型为"const std::weak_ptr"的操作数(或存在 不可接受 转换) test_cppunit_interpreter_base_multi_output c:\程序 文件 (x86)\Microsoft Visual Studio 14.0\VC\include\实用程序 216
看起来当Bar
位于命名空间中时找不到比较器......
我做错了什么吗?或者这可能是编译器错误?
应将operator==
移动到命名空间中以使 ADL 生效;ADL 还将检查用作模板参数的类型(即MyNamespace::Bar
),并将关联的命名空间(即MyNamespace
)添加到名称查找集中。 即
namespace MyNamespace
{
class Bar;
bool operator==( std::weak_ptr<Bar> left,
std::weak_ptr<Bar> right )
{
return left.lock() == right.lock();
}
}
为什么第一种情况工作正常?
因为 ADL 也适用于全局命名空间。对于第一种情况,Bar
和operator==
都在同一命名空间(即全局命名空间)中定义。
为什么第二种情况不起作用?
首先要注意的是,std::find
是在命名空间std
中定义的,并且其中定义了许多operator==
(具有不同的参数类型)。然后根据非限定名称查找的规则,当在命名空间std
找到operator==
时,名称查找停止。这意味着如果没有 ADL 的帮助,将根本找不到全局命名空间中定义的operator==
。
相关文章:
- 继承函数的重载解析
- 你能重载对象变量名本身返回的内容吗
- 从父命名空间重载类型
- 使用C++中的模板和运算符重载执行矩阵运算
- 为什么这个运算符<重载函数对 STL 算法不可见?
- 重载操作程序时出错>>用于类中的字符串 memebr
- 一个关于在C++中重载布尔运算符的问题
- 不同翻译单元中不可重载的非内联函数定义
- 为什么使用SFINAE而不是函数重载
- 为什么我不能在 C++ 中的特定函数重载中调用同一函数的任何其他重载?
- 将重载的成员函数传递给函数模板
- c++:可变模板和函数重载
- 重载元组索引运算符-C++
- 如何使用重载的相等(==)运算符向测试用例添加描述
- 重载==不适用于二进制树
- 为什么Mat类的两个对象可以在不重载运算符+的情况下添加
- 重载运算符new[]的行为取决于析构函数
- 正在尝试重载二进制搜索树分配运算符
- 重载Singly Linked List中的赋值运算符
- 取消引用运算符不能重载