为什么如果我不超载,我得到“不匹配‘运算符=’”自定义类?

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

我周围的一派,发现像this onethat one问题,但没有一个似乎有我同样的问题。搜索错误(no match for ‘operator=’)和说明(no known conversion for argument 1)产生一些结果。

我建立一个数据记录分析器,和类参数和表达需要彼此作为成员。我想我已经正确地实现向前声明,但我仍然有一个问题存在。下面,我尽最大努力最小重现错误。

parameter.h

#ifndef PARAMETER_H
#define PARAMETER_H


#include <iostream>
#include <string>
#include <vector>

class Expression;

class Parameter {
   public:
    explicit Parameter(std::string&);
    explicit Parameter(Expression*);
    std::string str;
    Expression* e;
    friend std::ostream& operator<<(std::ostream&, const Parameter&);
};

class Expression {
   public:
    Expression(Parameter&, std::string&, Parameter&);
    Parameter l;
    std::string op;
    Parameter r;
    friend std::ostream& operator<<(std::ostream&, const Expression&);
};

#endif  // PARAMETER_H

parameter.cpp

#include "parameter.h"

Parameter::Parameter(std::string& param) : str(param) {}

Parameter::Parameter(Expression* expr) { e = expr; }

std::ostream& operator<<(std::ostream& strm, const Parameter& p) {
    if (p.str.empty()) {
        // parameter is an expression
        strm << *p.e;
    } else {
        // parameter is ID or STRING
        strm << p.str;
    }

    return strm;
}

Expression::Expression(Parameter& left, std::string& oper, Parameter& right) : l(left), op(oper), r(right) {}

std::ostream& operator<<(std::ostream& strm, const Expression& e) {
    strm << "(" << e.l << " " << e.op << " " << e.r << ")";

    return strm;
}

main.cpp中

#include <iostream>
#include <string>
#include <vector>

#include "parameter.h"

int main() {
    std::string sample1 = "word";
    std::string sample2 = "another";
    std::string sample3 = "yep";

    std::vector<std::string> samples;
    samples.push_back(sample1);
    samples.push_back(sample2);
    samples.push_back(sample3);

    std::vector<Parameter> params;

    params.insert(params.end(), samples.begin(), samples.end());

    return 0;
}

我使用gcc version 7.3.0 (Ubuntu 7.3.0-27ubuntu1~18.04)。与g++ -Wall -Werror -std=c++17 -g *.cpp编译给出如下:

In file included from /usr/include/c++/7/bits/char_traits.h:39:0,
             from /usr/include/c++/7/ios:40,
             from /usr/include/c++/7/ostream:38,
             from /usr/include/c++/7/iostream:39,
             from recursiveDescent.h:4,
             from recursiveDescent.cpp:1:
/usr/include/c++/7/bits/stl_algobase.h: In instantiation of ‘static _OI 
std::__copy_move<false, false, 
std::random_access_iterator_tag>::__copy_m(_II, _II, _OI) [with _II = 
std::__cxx11::basic_string<char>*; _OI = Parameter*]’:
/usr/include/c++/7/bits/stl_algobase.h:386:44:   required from 
[...skipped...] required from here
/usr/include/c++/7/bits/stl_algobase.h:324:18: error: no match for ‘operator=’ (operand types are ‘Parameter’ and ‘std::__cxx11::basic_string<char>’)
    *__result = *__first;
    ~~~~~~~~~~^~~~~~~~~~
In file included from predicate.h:8:0,
                 from datalogProgram.h:8,
                 from recursiveDescent.h:8,
                 from recursiveDescent.cpp:1:
parameter.h:10:7: note: candidate: Parameter& Parameter::operator=(const Parameter&)
 class Parameter {
   ^~~~~~~~~
parameter.h:10:7: note:   no known conversion for argument 1 from ‘std::__cxx11::basic_string<char>’ to ‘const Parameter&’
parameter.h:10:7: note: candidate: Parameter& Parameter::operator=(Parameter&&)
parameter.h:10:7: note:   no known conversion for argument 1 from ‘std::__cxx11::basic_string<char>’ to ‘Parameter&&’
c++ g++
1个回答
0
投票

OP的构造Parameter::Parameter(std::string&)(或如PeterT指出,更好Parameter::Parameter(const std::string&))可以(而且会)用于std::string的隐式转换Parameter

params.insert(params.end(), samples.begin(), samples.end());

但OP通过使explicit明确禁止这一点。

我经常使用的构造函数的explicit可与一个说法,因为我怕意外的转换被称为(我想有控制权我的代码的错觉)。在这种情况下,它可能会更容易掉落explicit

样品:

#include <iostream>
#include <string>
#include <vector>

class Parameter {
  private:
    std::string _name;
  public:
    /*explicit*/ Parameter(const std::string &name): _name(name) { }
    const std::string& name() const { return _name; }
};

std::ostream& operator<<(std::ostream &out, const Parameter &param)
{
  return out << "Parameter '" << param.name() << "'";
}

template <typename T>
std::ostream& operator<<(std::ostream &out, const std::vector<T> &vec)
{
  const char *sep = "";
  for (const T &elem : vec) {
    out << sep << elem;
    sep = ", ";
  }
  return out;
}

int main()
{
  const std::vector<std::string> samples({ "word", "another", "yep" });
  std::vector<Parameter> params;
  params.insert(params.end(), samples.begin(), samples.end());
  std::cout << "params: " << params << '\n';
  return 0;
}

输出:

params: Parameter 'word', Parameter 'another', Parameter 'yep'

Live Demo on coliru

© www.soinside.com 2019 - 2024. All rights reserved.