不匹配'operator <

问题描述 投票:1回答:2

对不起,我复制了this question,但我没有在那里发表评论所需的声誉,而且那些答案对我来说并不令人信服。

#include<iostream>

class my_ostream : public std::ostream
{
    public:
    std::string prefix;

    my_ostream():prefix("*"){}

    my_ostream& operator<<(const std::string &s){
        std::cout << this->prefix << s;
        return *this;
    }
};

int main(){
  my_ostream s;
  std::string str("text");
  s << str << std::endl;
}

我得到:

在's.my_ostream :: operator <<中不匹配'operator <<'(((const std :: string&)((const std :: string *)(&str))))<< std :: endl'

而且我不明白为什么。如果它适用于ostream,它应该适用于my_ostream。这个程序有效:

#include <iostream>
using namespace std;

class a{};
class b:public a{};
class c:public b{};

void f(a){cout << 'a' << endl;}
void f(b){cout << 'b' << endl;}
void f(b, a){cout << "b, a" << endl;}
void f(c){cout << 'c' << endl;}
void f(c, int){cout << "c, int" << endl;}

void f(a*){cout << "pa" << endl;}
void f(b*){cout << "pb" << endl;}
void f(b*, a*){cout << "pb, pa" << endl;}
void f(c*){cout << "pc" << endl;}
void f(c*, int){cout << "pc, int" << endl;}

int main(){
  a ao; b bo; c co;
  f(ao); f(bo); f(co);
  f(co, ao);
  a *pa=new(a); b *pb=new(b); c *pc=new(c);
  f(pa); f(pb); f(pc);
  f(pc, pa);
  return 0;}

它输出:

a
b
c
b, a
pa
pb
pc
pb, pa

如此简单的重载并不能解释这个错误。另外,我不在这里介绍模板,因此未确定的模板类型参数不应起作用。阅读iostream代码证明非常困难,所以我很欣赏任何见解。

c++ iostream manipulators
2个回答
2
投票

简单的重载确实解释了这个错误。事实上,std::cout只会使问题复杂化。以下也不起作用:

int main(){
  my_ostream s;
  s << 1;
}

问题是你的operator <<重载实际上隐藏了为基类定义的所有重载。

粗略地说,C ++在范围解析后会重载分辨率。因此,C ++首先检查是否在类的范围内定义了operator <<。有!因此它停止在那里搜索更多通用函数,并且只考虑已经为重载解析找到的函数。唉,std::string只有一次重载,所以呼叫失败了。

这可以通过将operator <<定义为不是成员函数而是自由函数来解决:

my_ostream& operator<<(my_ostream& out, const std::string &s) {
    std::cout << out.prefix << s;
    return out;
}

...但当然这只能解决你的一些问题,因为你的类定义只是语义错误;你不能像这样子类化IO流。在这里,我的知识失败但我认为为了做你想做的事你应该覆盖流缓冲区的uflow函数。


0
投票

修复此问题的另一种方法是将运算符<< overloading'声明为朋友。

class my_ostream : public std::ostream
{
   public:
     std::string prefix;

   my_ostream():prefix("*"){}

   template <class T>
     friend my_ostream& operator<<(my_ostream& my_os, const T& s){
        std::cout << my_os.prefix << s;
     return my_os;
   }
};
© www.soinside.com 2019 - 2024. All rights reserved.