C++ |派生类正在访问基类的私有成员而不是它自己的私有成员

如何解决C++ |派生类正在访问基类的私有成员而不是它自己的私有成员

对不起,如果这是一个明显的问题,但我已经四处搜索,但我仍然不清楚如何解决这个问题。这是我的代码

#include <iostream>
#include <string>
using namespace std;

class PermMagnet {
public:
    string name;
    int ac_rating;
    int dc_rating;
    int mass_kg;
    int age;
    PermMagnet(){
        // default constructor
        name = "";
        ac_rating = 0; dc_rating = 0;
        mass_kg = 0; age = 0;
    }
    PermMagnet(string c_name,int c_ac_rating,int c_dc_rating,int c_mass_kg,int c_age){
        // parameterised constructor
        name = c_name;
        ac_rating = c_ac_rating;
        dc_rating = c_dc_rating;
        mass_kg = c_mass_kg;
        age = c_age;
    }
    string get_owner(){
        return owner;
    }
    string get_classifier(){
        return classifier;
    }
    int get_coil_count(){
        return coil_num;
    }
protected:
    string location = "facility hall";
private:
    string owner = "Unspecified Staff";
    string classifier = "MAG-DP-";
    const int coil_num = 2;
};

class ElecMagnet : public PermMagnet {
public:
    // inherit base class constructors
    using PermMagnet::PermMagnet;

    string get_location(){
        return location;
    }

private:
    string owner = "Specified Staff";
    string classifier = "MAG-QD-";
    const int coil_num = 4;
};

int main() {

    // Create object using default constructor
    PermMagnet perm1;
    cout << "'perm1' age: " << perm1.age << endl;

    // Create object using parameterised constructor
    PermMagnet perm2("PermMagnet 2",380,400,1500,35);
    cout << "'perm2' age: " << perm2.age << " | 'perm2' name: " << perm2.name << endl;
    cout << "Owner of 'perm2': " << perm2.get_owner() << endl;
    cout << "Upper current bound of 'perm2': " << perm2.get_current_limit("upper") << "A" << endl;
    cout << "Number of coils in 'perm2': " << perm2.get_coil_count() << endl << endl;

    // Create a ElecMagnet (derived class) object
    ElecMagnet elec1("ElecMagnet 1",170,200,850,27);
    cout << elec1.get_classifier() << endl;
    cout << elec1.get_coil_count() << endl;

    return 0;
}

这段代码输出是:

'perm1' age: 0
'perm2' age: 35 | 'perm2' name: PermMaget 2
Owner of 'perm2': Unspecified Staff
Upper current bound of 'perm2': 780A
Number of coils in 'perm2': 2

MAG-DP-
2

Process finished with exit code 0

如您所见,我希望“owner”、“classifier”和“coil_num”成为用户无法更改的私有成员。这些也因所讨论的课程而异。

问题:

问题是输出的最后两行。当派生类(ElecMagnet)继承返回这些成员的公共函数时,它返回基类的成员;不是自己的。您可以看到这一点,因为它返回 PermMagnet 类的“classifier”和“coil_num”。

有谁知道派生类为什么会这样?它不应该访问自己的私有成员而不是基础成员吗?

解决方法

私有成员只能被相同类或朋友的成员函数访问。例如,C++ 的行为不像 Python,后者根据当前指针 this (self) 动态执行变量搜索。

,

发生这种情况是因为你还在调用基类的方法,它无法知道派生类的成员变量,因此不会使用它们。 要归档您想要的内容,您需要在派生类中覆盖 get_classifier(),以便通过 ElecMagnet 引用调用它会调用派生方法。您可以选择使方法 virtual,甚至在通过 ElecMagnet 引用在 PermMagnet 值上调用它时调用派生版本。 (通常建议为继承而设计的类 virtual。)

或者,根据您的用例,您可以编写显式构造函数来调用(可能受保护的)基构造函数,而不是 using 基构造函数,并为 classifier 提供适当的值。这将完全节省派生类中的额外成员变量。

,

继承使用“是一个”关系。意思是,您应该能够说子对象“是”基础对象。如果你不能这样说,你就不应该使用继承。

这对于构建派生对象意味着它是一个复合体,包含您的基础对象以及通过继承添加的任何额外位。

您调用了一个基础对象函数,这意味着它将访问基础对象数据。在派生类中重复数据显然不起作用,而且是一种不好的做法。如果您需要相同的数据,请使用它,不要重复。

您可以通过让派生类构造函数始终首先在初始化部分调用基类构造函数来实现这一点。您根本不使用初始化部分,这可能会导致很多低效率并最终出现错误。

随机位: 所有类数据都应该是私有的。类的 protected: 部分包含您希望孩子可以公开访问但不想完全公开的任何内容。如果您不希望 getter 公开可用,他们可以去那里。我在这里放置了几个仅为 ElecMagnet 而存在的构造函数。 using namespace std; 是不好的做法。

代码:

#include <iostream>
#include <string>

class PermMagnet {
 public:
  PermMagnet() = default;
  PermMagnet(std::string c_name,int c_ac_rating,int c_dc_rating,int c_mass_kg,int c_age)
      : name(c_name),ac_rating(c_ac_rating),dc_rating(c_dc_rating),mass_kg(c_mass_kg),age(c_age) {}  // Use the initialization section

  std::string get_owner() const { return owner; }  // Mark getters as const

  std::string get_classifier() const { return classifier; }

  int get_coil_count() const { return coil_num; }

  std::string get_location() const { return location; }

  int get_age() const { return age; }

  std::string get_name() const { return name; }

 protected:
  PermMagnet(std::string owner,std::string classifier,int coilNum)
      : owner(owner),classifier(classifier),coil_num(coilNum) {}

  PermMagnet(std::string name,int ac_rating,int dc_rating,int mass_kg,int age,std::string owner,int coilNum)
      : name(name),ac_rating(ac_rating),dc_rating(dc_rating),mass_kg(mass_kg),age(age),owner(owner),coil_num(coilNum) {}

 private:
  std::string owner = "Unspecified Staff";
  std::string classifier = "MAG-DP-";
  const int coil_num = 2;  // const probably unnecessary here,but left it
  std::string location = "facility hall";
  std::string name = "";
  int ac_rating = 0;
  int dc_rating = 0;
  int mass_kg = 0;
  int age = 0;
};

class ElecMagnet : public PermMagnet {
 public:
  ElecMagnet() : PermMagnet("Specified Staff","MAG-QD-",4) {}

  ElecMagnet(std::string name,int age)
      : PermMagnet(name,ac_rating,dc_rating,mass_kg,age,"Specified Staff",4) {}

  // NO NEED FOR REPETITIVE PRIVATE SECTION
};

int main() {
  // Create object using default constructor
  PermMagnet perm1;
  std::cout << "'perm1' age: " << perm1.get_age()
            << '\n';  // Prefer '\n' over std::endl

  // Create object using parameterised constructor
  PermMagnet perm2("PermMagnet 2",380,400,1500,35);
  std::cout << "'perm2' age: " << perm2.get_age()
            << " | 'perm2' name: " << perm2.get_name() << '\n';
  std::cout << "Owner of 'perm2': " << perm2.get_owner() << '\n';
  std::cout << "Number of coils in 'perm2': " << perm2.get_coil_count()
            << "\n\n";

  // Create a ElecMagnet (derived class) object
  ElecMagnet elec1("ElecMagnet 1",170,200,850,27);
  std::cout << elec1.get_classifier() << '\n';
  std::cout << elec1.get_coil_count() << '\n';

  return 0;
}

请注意,ElecMagnet 类目前只是几个构造函数,您的主函数的行为符合预期。

版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。

相关推荐


Selenium Web驱动程序和Java。元素在(x,y)点处不可单击。其他元素将获得点击?
Python-如何使用点“。” 访问字典成员?
Java 字符串是不可变的。到底是什么意思?
Java中的“ final”关键字如何工作?(我仍然可以修改对象。)
“loop:”在Java代码中。这是什么,为什么要编译?
java.lang.ClassNotFoundException:sun.jdbc.odbc.JdbcOdbcDriver发生异常。为什么?
这是用Java进行XML解析的最佳库。
Java的PriorityQueue的内置迭代器不会以任何特定顺序遍历数据结构。为什么?
如何在Java中聆听按键时移动图像。
Java“Program to an interface”。这是什么意思?
Java在半透明框架/面板/组件上重新绘画。
Java“ Class.forName()”和“ Class.forName()。newInstance()”之间有什么区别?
在此环境中不提供编译器。也许是在JRE而不是JDK上运行?
Java用相同的方法在一个类中实现两个接口。哪种接口方法被覆盖?
Java 什么是Runtime.getRuntime()。totalMemory()和freeMemory()?
java.library.path中的java.lang.UnsatisfiedLinkError否*****。dll
JavaFX“位置是必需的。” 即使在同一包装中
Java 导入两个具有相同名称的类。怎么处理?
Java 是否应该在HttpServletResponse.getOutputStream()/。getWriter()上调用.close()?
Java RegEx元字符(。)和普通点?