简体   繁体   English

C ++重载运算符,具有来自模板的多重继承

[英]C++ overloading operator with multiple inheritance from templates

I have an hierarchy that represents some part of an HTTP client and looks like this: 我有一个层次结构,代表HTTP客户端的某些部分,如下所示:

typedef list<pair<string, string> > KeyVal;
struct Header { string name; string value; ...};
struct Param { string name; string value; ...};

/* Something that contains headers */
template<typename T> class WithHeaders {
  KeyVal headers;
public:
  virtual T &operator <<(const Header &h) {
    headers.push_back(pair<string, string>(h.name, h.value));
    return static_cast<T&> (*this);
  }
};

/* Something that contains query params */
template<class T> class WithQuery {
    KeyVal query_params;

public:
    virtual T &operator <<(const Param &q) {
      query_params.push_back(pair<string, string>(q.name, q.value));
      return static_cast<T&> (*this);
    }

    const KeyVal &get_query() const {return query_params;}
};

/* Http Request has both headers and query parameters */
class Request: public WithQuery<Request>, public WithHeaders<Request> {...};

So that I expected to be able to do things like request << Header(name, value) << Param("page", "1") (and later will reuse WithHeaders in the corresponding Response class). 所以我希望能够执行诸如request << Header(name, value) << Param("page", "1")类的WithHeaders (后来将在相应的Response类中重用WithHeaders )。

The code that I'm trying to compile is: 我正在尝试编译的代码是:

Request rq = Request("unused", "unused", "unused");
rq << Header("name", "value");

However, I get: 但是,我得到:

test/test_client.cpp:15:30: error: request for member ‘operator<<’ is ambiguous
In file included from test/test_client.cpp:1:0:
test/../client.h:45:16: error: candidates are: 
    T& WithQuery<T>::operator<<(const Param&) [with T = Request]         
    T& WithHeaders<T>::operator<<(const Header&) [with T = Request]

I must be missing something, but it seems quite easy to distinguish Param from Header during the compilation time. 我必须遗漏一些东西,但在编译期间似乎很容易区分ParamHeader So, the questions are: 所以,问题是:

  • why does it fail and how to fix it? 为什么会失败以及如何解决?
  • is that a reasonable thing to do at all or there's a simpler design? 这是一个合理的事情还是设计更简单?

I believe it should work, so it's most likely a GCC bug. 我相信它应该有用,所以它很可能是一个GCC错误。 As pointed out by @refp in the comments, the lookup is actually ambiguous and GCC is correct in rejecting it. 正如@refp在评论中指出的那样,查找实际上是不明确的,而GCC在拒绝它时是正确的。

This is how you make it work: 这就是你如何使它工作:

class Request: public WithQuery<Request>, public WithHeaders<Request> {
public:
    using WithHeaders<Request>::operator<<;
    using WithQuery<Request>::operator<<;
};

Live example 实例

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM