gpt4 book ai didi

c++ - 关于 C++ 中模板上重载运算符的一些编译器错误

转载 作者:行者123 更新时间:2023-11-28 08:26:02 26 4
gpt4 key购买 nike

我有一些代码有一些错误,我根本不知道如何修复。我已经问过我的教授和助教,除了更准确地理解错误的含义之外,还没有运气地咨询了互联网。据我所知,编译器要么将我的重载运算符与内置运算符混淆,要么根本不将其识别为重载运算符。

我收到以下错误和警告:

||=== project 4, Debug ===|
\project 4\forest.h|13|warning: friend declaration 'Forest<NODETYPE>& operator+(Forest<NODETYPE>&, Forest<NODETYPE>&)' declares a non-template function|

\project 4\forest.h|13|note: (if this is not what you intended, make sure the function template has already been declared and add <> after the function name here) |

\project 4\forest.h|14|warning: friend declaration 'std::ostream& operator<<(std::ostream&, const Forest<NODETYPE>&)' declares a non-template function|

\project 4\forest.h|15|warning: friend declaration 'std::istream& operator>>(std::istream&, Forest<NODETYPE>&)' declares a non-template function|

\project 4\main.cpp||In function 'int main()':|
\project 4\main.cpp|21|error: ambiguous overload for 'operator>>' in 'file >> intForest'|

c:\program files (x86)\codeblocks\mingw\bin\..\lib\gcc\mingw32\4.4.1\include\c++\istream|119|note: candidates are: std::basic_istream<_CharT, _Traits>& std::basic_istream<_CharT, _Traits>::operator>>(std::basic_istream<_CharT, _Traits>& (*)(std::basic_istream<_CharT, _Traits>&)) [with _CharT = char, _Traits = std::char_traits<char>] <near match>|

c:\program files (x86)\codeblocks\mingw\bin\..\lib\gcc\mingw32\4.4.1\include\c++\istream|123|note: std::basic_istream<_CharT, _Traits>& std::basic_istream<_CharT, _Traits>::operator>>(std::basic_ios<_CharT, _Traits>& (*)(std::basic_ios<_CharT, _Traits>&)) [with _CharT = char, _Traits = std::char_traits<char>] <near match>|

c:\program files (x86)\codeblocks\mingw\bin\..\lib\gcc\mingw32\4.4.1\include\c++\istream|130|note: std::basic_istream<_CharT, _Traits>& std::basic_istream<_CharT, _Traits>::operator>>(std::ios_base& (*)(std::ios_base&)) [with _CharT = char, _Traits = std::char_traits<char>] <near match>|
||=== Build finished: 1 errors, 3 warnings ===|

当我尝试编译我的代码时。相关代码段如下:

(在 forest.h 中)

template< typename NODETYPE > class Forest
{

public:
friend Forest<NODETYPE>& operator+(Forest<NODETYPE>&, Forest<NODETYPE>&);
friend ostream& operator<<(ostream&, const Forest<NODETYPE>&);
friend istream& operator>>(istream&, Forest<NODETYPE>&);

Forest();
Forest( const Forest& otherForest);
~Forest();
void nodes(int&) const;

private:
ForestNode<NODETYPE> *root;

ForestNode<NODETYPE> *getNewNode( const NODETYPE &);
};

(在 forest.cpp 中)

template<typename NODETYPE> istream& operator>>(istream& file, const Forest<NODETYPE>& f1)
{
istream file2 = file;
int nodeCount = 0;
string blah = ' ';

while(getline(file2,blah))
{
nodeCount++;
}

ForestNode<NODETYPE> *forestNodeArray[nodeCount];//holds pointers to last node of depth i

getline(file,*f1.root.tag);
forestNodeArray[0] = &(*f1.root);
inputHelper(file, 0, *f1.root, forestNodeArray, nodeCount);

return file;
}

(最后,在 main.cpp 中)

int main()
{

Forest < char > intForest();
filebuf fb;
fb.open ("forest1.txt",ios::in);
istream file(&fb);


cout << typeid(intForest).name() << endl;
cout << typeid(file).name() << endl;

file >> intForest;

fb.close();


}

如有任何帮助,我们将不胜感激。

编辑:感谢 alex 和 alf,我明白了为什么不将它们视为模板函数。回想起来很明显,我只是对那些签名感兴趣。无论如何,我仍然得到关于模糊运算符的错误。为什么编译器不识别我的运算符并使用它,而不是尝试在 3 个内置版本的运算符之间做出决定,这些版本不可能有一个参数作为 Forest?

最佳答案

第二个错误在这一行:

Forest < char > intForest();

起初这可能令人惊讶,但该行并未声明 Forest<char> 类型的变量,而是一个不带参数并返回 Forest<char> 的函数.只需从声明中删除括号:

Forest < char > intForest;

关于第一个警告,这已经解释过了(声明为 friend 的函数不是模板,这意味着您必须手动为用于实例化 Forest<> 的每种类型实现它(可能您不t 想要那个)。另请注意,声明模板化的 operator+ 然后使模板成为@Alf P. Steinbach 答案中的 friend 意味着 Forest<int> 将成为 Forest<double> 的 friend ,这可能不是什么你需要。评论中的@Alex 建议只会使模板的特定实例成为 friend ,这可能更接近你想要的,但你需要在模板类之前声明模板化运算符,为此你需要转发声明模板类...

模板中无友元函数的常见模式是就地定义函数:

template <typename T>
class Forest {
// ...
friend Forest& operator+( Forest const & lhs, Forest const & rhs ) [1]
{
// implementation here
}
}
// [1] Note that the arguments are const references (read only), and also note that
// you do not need to provide the type argument inside the template declaration

这允许您将其定义为非模板函数,同时让编译器为您实例化该函数。此外,在处理模板时,在类定义中内联定义成员方法通常更简单。它让生活变得更简单,毕竟在大多数情况下,您确实需要在(相同的)头文件中提供实现。

然而,在定义二元运算符时,更好的方法是定义 operator+=作为成员方法,然后你可以很容易地定义operator+作为非 friend 免费功能。模式将是:

struct test {
test& operator+=( test const & );
};
test operator+( test lhs, test const & rhs ) { // [2]
lhs += rhs;
return lhs;
}
// [2] Note that the first argument is by value, and the second by const reference
// This means that the compiler will copy the first argument for you, and you
// are free to modify it with the += operator and return the copy.

现在,最棘手的部分是混合前两个建议。能够定义 operator+这是模板定义中的一个自由函数,一个常见的技巧是使它成为 friend ,即使出于访问原因不需要这样做:

template <typename T>
class Forest {
Forest& operator+=( Forest const & ) {
// implemenation
}
friend Forest operator+( Forest lhs, Forest const & rhs ) { // [3]
return lhs+=rhs;
}
};
// [3] friendship only to allow us to define the free function inside the template
// declaration.

关于c++ - 关于 C++ 中模板上重载运算符的一些编译器错误,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/4039979/

26 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com