我似乎无法找到这个新手问题的答案。如果我有课 //头文件(.h)
Class X {
public:
friend bool operator==(const X&, const X&);
inline size_type rows() const;
};
...等 当我去实现X的.cpp文件时,我是否应该包含内联和单词朋友在.cpp文件中的函数名称中。即,我应该实现类似于下面的文件
// CPP file (.cpp)
#include "X.h"
friend bool operator==(const X&, const X&) {
//implementation goes here
//return true/false
}
inline size_type rows() const {
return r;
}
或者我不应该包括这些,例如下面的
#include "X.h"
bool operator==(const X&, const X&) { ... }
size_type rows() const { ... }
答案 0 :(得分:6)
不,你不应该,即第二个版本是正确的。
friend
只能在类定义中使用(它没有任何意义),编译器会查找函数签名,以查找在类中声明的friend
函数的定义X
。
inline
用在头文件中(虽然,我猜,从技术上讲,也可以在cpp文件中使用它,只是没有意义)。请注意inline
只有在实际定义函数时才有意义 - 如果在cpp文件中单独提供函数定义,它就没有效果。
inline
的要点是提示编译器可以内联所讨论的函数(但不能保证 - 编译器可以自由决定是否内联函数)。如果函数是内联的,则对函数体的任何调用都将替换为函数体的副本。这实际上总是为了性能优势而做,以节省函数调用的成本,但代价是可能增加程序大小。现在,如果我们想要内联函数,我们希望它在所有地方内联,而不仅仅是在一个编译单元中;这就是为什么在实现文件中使用inline
关键字没有意义。
答案 1 :(得分:4)
定义friend
函数时,您可以使用以下两种选项之一:
在类定义
中定义friend
函数
namespace Test {
class test {
int priv;
friend void foo( test const & t ) { std::cout << t.priv << std::endl; }
};
}
在封闭的命名空间中定义它们:
namespace Test {
class test {
int priv;
friend void foo( test const & t );
};
void foo( test const & t ) {
std::cout << t.priv << std::endl;
}
}
但请注意,查找方面存在差异。特别是,第一种情况更具限制性(因此应该是首选的)因为只有ADL(Argument Dependent --aka Koening-- Lookup)会找到它,而在第二种情况下,只要考虑该命名空间,就会考虑该函数。
也就是说,为了让编译器将第一个函数视为重载,调用地点的参数必须为test
,而在第二种情况下,它将被视为标识符匹配时,如果参数可转换到test
,则可视为有效选项。
答案 2 :(得分:0)
您不希望函数体上的朋友前缀,只需要标题中的签名。