当前位置: 首页 > 编程笔记 >

C/C++ 公有继承、保护继承和私有继承的对比详解

相野
2023-03-14
本文向大家介绍C/C++ 公有继承、保护继承和私有继承的对比详解,包括了C/C++ 公有继承、保护继承和私有继承的对比详解的使用技巧和注意事项,需要的朋友参考一下

C/C++ 公有继承、保护继承和私有继承的区别

   在c++++的继承控制中,有三种不同的控制权限,分别是public、protected和private。定义派生类时,若不显示加上这三个关键字,就会使用默认的方式,用struct定义的类是默认public继承,class定义的类是默认private继承。这和Java有很大的不同,Java默认使用public继承,而且只有公有继承。

        1.使用public继承时,派生类内部可以访问基类中public和protected成员,但是类外只能通过派生类的对象访问基类的public成员。

        (1)基类的public成员在派生类中依然是public的。

        (2)基类中的protected成员在派生类中依然是protected的。

        (3)基类中的private成员在派生类中不可访问。

        2.使用protected继承时,派生类内部可以访问基类中public和protected成员,并且类外也不能通过派生类的对象访问基类的成员(可以在派生类中添加公有成员函数接口间接访问基类中的public和protected成员)。

        (1)基类的public成员在派生类中变为protected成员。

        (2)基类的protected成员在派生类中依然是protected成员。

        (3)基类中的private成员在派生类中不可访问。

       3.使用private继承时,派生类内部可以访问基类中public和protected成员,并且类外也不能通过派生类的对象访问基类的成员(可以在派生类中添加公有成员函数接口间接访问基类中的public和protected成员)。

        (1)基类的public成员在派生类中变成private成员。

        (2)基类的protected成员在派生类中变成private成员。

        (3)基类的private成员在派生类中不可访问。

        为了便于理解,我们用一个表格来说明这几种控制符使用的情况:  

派 生 方 式  基类的public成员 基类的protected成员 基类的private成员
public派生 还是public成员 变为protected成员 不可见
protected派生 变成protected成员    还是protected成员 不可见
private派生 变为private成员 变成private成员 不可见

       下面用代码简单叙述

#include <iostream> 

class Base { 
public: 
  int public_a; 
  virtual void test() = 0; 

protected: 
  int protected_a; 

private: 
  int private_a; 
}; 

//公有继承 
class PublicDerived : public Base { 
public: 
  virtual void test() { 
    public_a = 1;          //public_a public继承后还是public类型 
    protected_a = 2;        //protected_a 还是protected类型 
    //private_a = 3;         //派生类不可访问基类私有成员 
  } 
}; 

//保护继承 
class ProtectedDerived : protected Base { 
public: 
  virtual void test() { 
    public_a = 1;          //public_a protected继承后变为protected类型 
    protected_a = 2;        //protected_a 还是protected类型 
    //private_a = 3;        //派生类不可访问基类私有成员 
  } 
}; 

//私有继承 
class PrivateDerived : private Base { 
public: 
  virtual void test() { 
    public_a = 1;          //public_a private继承后变为private类型 
    protected_a = 2;        //protected_a private继承后变为private类型 
    //private_a = 3;        //派生类不可访问基类私有成员 
  } 
}; 

        在c++中public继承是is-a的关系。也就说适用于基类身上的一定也适用于派生类身上,因为每一个派生类对象也都是一个基类对象。派生类对象转换为基类对象在需要的时候是可以自动转化的。

        其实,protected和private派生后的派生类不是基类的子类,因为此派生类不能做基类所能做的所有事情。我们看看下面的代码

#include <iostream> 

class Person { 
public: 
  Person(){}; 

  void eat() {std::cout << "eat\n";} 
}; 

//公有继承 
class PublicStudent : public Person { 
public: 
  PublicStudent() {} 

  void study() {std::cout << "study\n";} 
}; 

//保护继承 
class ProtectedStudent : protected Person { 
public: 
  ProtectedStudent() {} 

  void study() {std::cout << "study\n";} 
}; 

//私有继承 
class PrivateStudent : private Person { 
public: 
  PrivateStudent() {} 

  void study() {std::cout << "study\n";} 
}; 

 

void func_test(Person &p) { 
  p.eat(); 
} 

int main() { 

  PublicStudent public_s; 
  ProtectedStudent protected_s; 
  PrivateStudent private_s; 

  func_test(public_s);          //正确,公有继承时,派生类可以做基类的所有事情 
  func_test(protected_s);         //出错,保护继承时,派生类不能做基类的所有事情 
  func_test(private_s);          //出错,私有继承时,派生类不可以做基类的所有事情 

  system("pause"); 

  return 0; 
} 

        func_test()需要一个Person类型的对象,在调用func_test(public_s)时实际传递的是PublicStudent对象,因为PublicStudent共有继承Person类,所以PublicStudent对象可以使用Person类中所有的公有成员,也就是说基类对象可以做的事,公有继承的派生类对象照样可以做。而ProtectedStudent和PrivateStudent都是非公有有继承,他们的对象不可以直接访问Person类的成员。但是在他们派生类的对象空间中是包含基类的的对象,只是无法让他们公开访问。从上面的代码可以看出来,protected继承和private继承的派生类已经不是基类的子类了。

       写了这么多,其实非公有继承很少遇到。

 感谢阅读,希望能帮助到大家,谢谢大家对本站的支持!

 类似资料:
  • FAQs in section [24]: [24.1] 如何表示“私有继承”? [24.2] 私有继承和组合(composition)有什么类似? [24.3] 我应该选谁:组合还是私有继承? [24.4] 从私有继承类到父类需要指针类型转换吗? [24.5] 保护继承和私有继承的关系是什么? [24.6] 私有继承和保护继承的访问规则是什么? 24.1 如何表示“私有继承”? 用 : priv

  • 本文向大家介绍详解C++编程中的私有继承和公有继承,包括了详解C++编程中的私有继承和公有继承的使用技巧和注意事项,需要的朋友参考一下 C++类的私有继承 在声明一个派生类时将基类的继承方式指定为private的,称为私有继承,用私有继承方式建立的派生类称为私有派生类(private derived class ), 其基类称为私有基类(private base class )。 私有基类的公用成

  • 问题内容: 假设Java具有以下层次结构类: 这是C#中相同代码的(盲)重复: 当我执行Java代码时,我得到了C#返回的信息。 对我来说,C#的结果更有意义,因为引用B调用了它自己的方法。 Java设计者决定打印而不是打印的逻辑是什么?我的意思是,为什么引用B在C中使用覆盖方法?这种方法的优势是什么? 如何更改Java代码以像C#一样打印出来?我的意思是,我怎么教Java调用它使用的完全引用的方

  • 主要内容:1、基类和派生类,2、多重继承继承与 封装和 多态被统称为面向对象编程的三大特性,本节我们主要来介绍一下继承这一特性。 在创建一个新类时,我们可以使用这个新定义的类继承一个已有的类,通过继承可以在创建新类时重用、扩展和修改被继承类中定义的成员。被继承的类称为“基类(父类)”,继承基类的类称为“派生类(子类)”。 需要注意的是,C# 中只支持单继承,也就是说一个派生类只能继承一个基类,但是继承是可以传递的,例如 ClassC 继

  • 面向对象程序设计中最重要的一个概念是继承。继承允许我们依据另一个类来定义一个类,这使得创建和维护一个应用程序变得更容易。这样做,也达到了重用代码功能和提高执行时间的效果。 当创建一个类时,您不需要重新编写新的数据成员和成员函数,只需指定新建的类继承了一个已有的类的成员即可。这个已有的类称为基类,新建的类称为派生类。 继承代表了 is a 关系。例如,哺乳动物是动物,狗是哺乳动物,因此,狗是动物,等

  • 主要内容:多继承下的构造函数,命名冲突在前面的例子中,派生类都只有一个基类,称为 单继承(Single Inheritance)。除此之外, C++也支持 多继承(Multiple Inheritance),即一个派生类可以有两个或多个基类。 多继承容易让代码逻辑复杂、思路混乱,一直备受争议,中小型项目中较少使用,后来的 Java、 C#、 PHP 等干脆取消了多继承。 多继承的语法也很简单,将多个基类用逗号隔开即可。例如已声明了类A