盐城网站定制seo优化方案报价
目录
1.隐藏
1.1隐藏的概念
1.2隐藏的两种方式
2.继承与友元
3、继承与静态成员
4.单继承和多继承
4.1单继承
4.2多继承
5.菱形继承
问题1:冗余性
问题2:二义性
6.虚拟继承
7.总结
1.隐藏
1.1隐藏的概念
在 C++ 中,继承是一种机制,使得子类可以继承父类的成员变量和成员函数。然而,当子类中出现和父类同名的成员变量或成员函数时,会发生一种特殊的现象,即隐藏。
隐藏是指:如果子类中出现了与父类同名的成员变量或成员函数,则子类中的这个成员会“隐藏”父类中的同名成员,使得父类中的同名成员在子类中不可见。
1.2隐藏的两种方式
成员变量隐藏:顾名思义是类中的成员函数被隐藏了
class Parent {
public:int W;
};class Child : public Parent {
public:int W; // 此处的 W 会隐藏 Parent 中的 W
};int main() {Child c;c.W = 10; // 此处修改的是 Child 中的 W,而不是 Parent 中的 Wreturn 0;
}
成员函数隐藏:和成员变量隐藏一样,成员函数被隐藏
class Parent {
public:void func() {cout << "Parent::func()" << endl;}
};class Child : public Parent {
public:void func() {cout << "Child::func()" << endl;}
};int main() {Child c;c.func(); // 此处调用的是 Child 中的 func(),而不是 Parent 中的 func()return 0;
}
2.继承与友元
在C++中,友元关系不能被继承,因为友元关系是独立于类定义的,并不是类的成员。因此,如果在父类中声明了一个友元函数或友元类,子类无法继承这种关系。
class Student;class Person{public:friend void Display(const Person& p, const Student& s);protected:string _name; //
姓名};class Student : public Person{protected:int _stuNum; //
学号};void Display(const Person& p, const Student& s){cout << p._name << endl;cout << s._stuNum << endl;}int main(){Person p;Student s;//
编译报错:
error C2248: “Student::_stuNum”:
⽆法访问protected
成员//
解决⽅案:
Display
也变成
Student
的友元即可Display(p, s);return 0
3、继承与静态成员
在 C++ 中,类可以包含静态成员变量和静态成员函数,其中静态成员变量属于类本身,而不是类的某个对象,因此它们可以在不创建类对象的情况下被访问。
当一个类继承另一个类时,子类可以继承其父类的静态成员,并且可以在子类中重新定义这些静态成员。在这种情况下,子类和父类各自拥有自己的静态成员变量,它们之间没有任何关系。
#include<iostream>
using namespace std;class Parent {
public:static int a;
};int Parent::a = 1000; //静态成员的定义只能在类外进行class Child : public Parent {
public:static int a; //类中只能声明静态成员
};int Child::a = 2000; //静态成员的定义只能在类外进行int main() {cout << Parent::a << endl; //输出1000cout << Child::a << endl; //输出2000return 0;
}
此外,静态成员函数也可以继承,并且可以在子类中重新定义。在子类中重新定义父类的静态成员函数时,子类的静态成员函数会隐藏父类的静态成员函数,因此如果在子类中需要调用父类的静态成员函数,需要使用作用域运算符 :: 来显式地调用。
4.单继承和多继承
4.1单继承
单继承是指一个类只能继承自一个基类。这意味着派生类只能有一个直接的基类。
如下代码所示:
class Base {
public:void baseFunction() {cout << "Base function" << endl;}
};class Derived : public Base {
public:void derivedFunction() {cout << "Derived function" << endl;}
};
4.2多继承
多继承允许一个类继承自多个基类。这意味着派生类可以有多个直接的基类。
class Base1 {
public:void base1Function() {cout << "Base1 function" << endl;}
};class Base2 {
public:void base2Function() {cout << "Base2 function" << endl;}
};class Derived : public Base1, public Base2 {
public:void derivedFunction() {cout << "Derived function" << endl;}
};
5.菱形继承
C++中的菱形继承是指在类的继承关系中,存在两个或更多个直接或间接的基类,它们之间形成了一个类似菱形的结构。这种继承结构通常出现在多层继承中,当一个派生类同时从两个不同的基类继承到了同一个基类时,就可能导致问题。
问题1:冗余性
主要体现在代码的重复。在菱形继承中,派生类会继承两个基类的所有公共和私有成员。如果这些成员在两个基类中定义了相同的实现,那么在派生类中可能会有重复的代码,这不仅增加了代码量,还可能导致维护困难,因为需要在所有相关的实现中同步更新。
问题2:二义性
二义性是指在菱形继承的情况下,派生类可能会有两个或更多的基类提供了相同的函数或数据成员,这在调用时会导致编译器无法确定调用哪个版本。例如,如果基类A和B都有一个同名的函数,而在派生类中没有明确指定调用哪一个,就会产生二义性错误。
class Person
{
public :string _name ; // 姓名
};
class Student : public Person
{
protected :int _num ; //学号
};
class Teacher : public Person
{
protected :int _id ; // 职工编号
};
class Assistant : public Student, public Teacher
{
protected :string _majorCourse ; // 主修课程
};
void Test ()
{// 这样会有二义性无法明确知道访问的是哪一个Assistant a ;
a._name = "peter";
// 需要显示指定访问哪个父类的成员可以解决二义性问题,但是数据冗余问题无法解决a.Student::_name = "xxx";a.Teacher::_name = "yyy";
}
6.虚拟继承
下面我们来讲解一种解决上面问题的方法——虚拟继承
虚继承是一种特殊的继承方式,用于解决菱形继承中的冗余性和二义性问题
class BaseClass {
public:int var;
};class LeftChild : public virtual BaseClass {
public:// ...
};class RightChild : public virtual BaseClass {
public:// ...
};class FinalChild : public LeftChild, public RightChild {
public:// ...
};
在上面的示例中,LeftChild 和 RightChild 都virtually继承自 BaseClass,这样在 FinalChild 继承 LeftChild 和 RightChild 时,就不会再继承 BaseClass 的两份副本,避免了冗余性问题。此时,BaseClass 的成员变量 var 在 FinalChild 中只有一份,并且不会发生二义性问题。
总之,C++ 通过虚继承解决了菱形继承中的冗余性和二义性问题,使得在使用继承时更加灵活和安全。
7.总结
以上是C++继承博主学习的全部内容,可能不全但是覆盖了继承的几乎知识点,后期主播会积极补充有关继承其他有关知识点,谢谢大家的关注和支持!