C++ 基类和派生类
- 格式:doc
- 大小:27.00 KB
- 文档页数:4
C++语法总结(一)1. I/O流的常用控制符dec 置基数为10hex 置基数为16oct 置基数为8setfill(c) 设填充字符为Csetprecision(n) 设显示小数精度为n位setw(n) 设域宽为N个字符setiosflags(ios::fixed) 固定的符点显示setiosflags(ios::scientific)指数表示setiosflags(ios::left) 左对齐setiosflags(ios::right) 右对齐setiosflags(ios::skipws) 忽略前导空白setiosflags(ios::uppercase) 16进制数大写输出setiosflags(ios::lowercase) 6进制数小写输出setiosflags(ios::showpoint) 显示小数点setiosflags(ios::showpos) 正数前面加上正号2.头文件:*iostream.h* *iomanip.h* *stdlib.h*cout/cin 流的控制符 exit(0)3.指针的几种类型:int (*p)();p为指向函数的指针变量,该函数带回一个整形值int *p();p为带回一个指针的函数,该指针指向整形数据int (**)[n]p为一个指向一个指针变量的指针变量,被指向的指针变量指向一个含n个整形数据的一维数组4.构造函数和析构函数特点及其区别:a.构造函数可以有任意个形参,还可以重载(多个参数个数不同的函数);但析构函数不能有形参,因为是系统自动调用的.b.构造函数不可以任意调用,只准系统调用;而析构函数不仅系统调用,也可以任意调用.5.构造函数和析构函数什么时候自动运行?(例61)构造函数:一般在定义类对象时自动运行.析构函数:如果一个函数中定义了一个对象,则在这个函数运行结束时就执行一次;当一个对象是使用NEW运算符被动态创建的,在使用DELETE运算符释放它时,DELETE将会自动调用析构函数.拷贝初始化构造函数:当用tpoint N(M);语句时调用一次;当对象作为实参向形参传递时,即对形参初始化时执行一次;当遇到M=return(N);语句,即对M进行初始化时调用一次;6. this 指针用法:例63中,当程序执行语句list elem(i);时,系统对this指针进行了如下的缺省赋值:this=&list;赋值成员函数举例(此例在例63基础上):void Assign(linear_list&);//说明语句;void linear_list::Assign(linear_list&p){if(&p==this)return;nmax=p.nmax;nelem=p.nelem;list=new int[nmax];for(int i=0;i<nmax;i++)list[i]=p.list[i];}7.const,volatile对象和const,volatile成员函数格式:const person per/volatile person per;int func(int) const;/char func(int) volatile;说明:1.普通对象既可以访问const对象,又可以访问volatile对象;2.const对象只能访问用const修饰的成员函数;volatile对象只能访问用其所长volatile修饰的成员函数;3.也可以同时说明为const volatile对象或const volatile成员函数;const volatile对象只能访问const volatile成员函数;const/volatile对象也能访问const volatile成员函数;8.不同继承方式的基类和派生类特性------------------------------------------------继承方式 | 基类特性 | 派生类特性------------------------------------------------| public | public公有继承 | protected | protected| private | 不可访问------------------------------------------------| public | private私有继承 | protected | private| private | 不可访问------------------------------------------------| public | protected保护继承 | protected | protected| private | 不可访问------------------------------------------------A:帮助理解:1)对于公有继承方式:a.基类成员对其对象的可见性:公有成员可见,其他不可见。
面向对象程序设计一、选择填空1. 关于C + +与C语言的关系描述中,(D)是错误的。
(A)C语言是C + +的一个子集 (B) C语言与C++是兼容的(C) C++对C语言进行了一些改进(D) C++和C语言都是面向对象的2. 下面关于对象概念的描述中,(B)是错误的。
(A )对象就是C语言的结构变量(B)对象代表着正在创建的系统中的一个实体(C)对象是一个状态的操作(或方法)的封装体(D)对象之间的信息传递是通过消息进行的3. 下面关于类概念的描述中,(D)是错误的。
(A )类是抽象数据类型的实现(B )类是具有共同行为的若干对象的统一描述体(C)类是创建对象的样板(D)类就是C语言的结构类型4. 按照标识符的要求,(A )符号不能组成标识符。
(A )连接符 (B)下划线 (C)大小写字母 (D)数字字符5. 下列符号中,(C)不可作为分隔符。
(A), (B) : (C)? ( D);6. 在16位机中,int型字宽为(A)字节。
(A ) 2 ( B) 4 ( C) 6 (D) 87. 类型修饰符unsigned修饰(D)类型是错误的。
(A)char ( B) int (C) long int (D) float& 对于int *pa[5];的描述,(D)是正确的。
(A)pa是一个指向数组的指针,所指向的数组是5个int型元素(B)pa是一个指向某数组中第5个元素的指针,该元素是int变量(C)pa[5]表示某个数组的第5个元素(D)pa是一个具有5个元素的指针数组,每个元素是一个int型指针。
9. 下列关于指针的运算中,(D)是非法的。
(A)两个指针在一定条件下,可以进行相等或不等的运算。
(B)可以有一个空指针赋值给某个指针。
(C)一个指针可以加上一个整数。
(D)两个指针在一定条件下,可以相加。
10. 指针可以用来表示数组元素,下列表示中( A )是错误的。
已知:int a[3][7];(A)*(a+1)[5] ( B) *(*a+3) (C) *(*(a+1)) (D) *(&a[0][0]+2)11. 下列表示引用的方法中,(A )是正确的。
C++作业答案第7章继承7.1 选择题1.在c++中,类与类之间的继承关系具有( c )。
(a) 自反性 (b) 对称性 (c) 传递性 (d) 反对称性2.下列关于类的继承描述中,( a,b )是错误的。
(a) 派生类可以访问基类的所有数据成员,调用基类的所有成员函数(b) 派生类也是基类,所以基类具有派生类的全部属性和方法(c) 继承描述类的层次关系,派生类可以具有与基类相同的属性和方法(d) 一个基类可以有多个派生类,一个派生类可以有多个基类3.当一个派生类公有继承一个基类时,基类中的所有公有成员成为派生类的( a )。
(a) public 成员 (b)private成员(c) protected成员 (d)友员4.当一个派生类私有继承一个基类时,基类中的所有公有成员和保护成员成为派生类的( b )。
(a) public 成员 (b)private成员(c) protected成员 (d)友员5.当一个派生类保护继承一个基类时,基类中的所有公有成员和保护成员成为派生类的( c )。
(a) public 成员 (b)private成员(c) protected成员 (d)友员6.不论派生类以何种方式继承基类,都不能使用基类的( b )。
(a) public 成员 (b)private成员(c) protected成员 (d)public 成员和protected成员7.下面描述中,错误的是( b, c )。
(a) 在基类定义的public成员在公有继承的派生类中可见,也能在类外被访问(b) 在基类定义的public和protected成员在私有继承的派生类中可见,在类外可以被访问(c) 在基类定义的public和protected成员在保护继承的派生类中不可见(d) 在派生类中不可见的成员要变成可访问的需进行访问声明8.在c++中,不能被派生类继承的函数是( b,c )。
(a) 成员函数 (b)构造函数(c) 析构函数 (d)静态成员函数9.在创建派生类对象时,构造函数的执行顺序是( d )。
C++中三种继承⽅式的区别还可参考另⼀篇⽂章:public公有继承protected保护继承private我们知道类的private和protected成员,在类外是不可以使⽤的.只有public成员可以在类外直接使⽤.公有继承时,基类的private成员派⽣类也不可⽤,基类的public和protected成员在派⽣类中可直接使⽤.继承过来(变成派⽣类相应的public和protected成员)只有public成员在派⽣类外可以直7a64e58685e5aeb931333262363034接使⽤.保护继承时,基类的private成员仍为有私有.基类的public和protected成员变成派⽣类的protected成员,这时在派⽣类外也不能直接使⽤原基类的public成员时,基类的private成员仍为有私有.基类的public和protected成员将变成派⽣类的private成员.举个例⼦.class A{public:int m_nTelNum;protected:int m_nAge;private:int m_nMoney;};class B:public A{void SetTelNum(int nTelNum){m_nTelNum=nTelNum;}void SetAge(int nAge){m_nAge=nAge;}void SetMoney(int nMoney){m_nMoney=nMoney;//这⾥就出现错误,因为基类的private成员不能⽤.};B objB;//创建B类的对象objBobjB.m_nTelNum=123456;//可以objB.m_nAge=30;//错误.public继承中基类的protected在派⽣类中是protectedobjB.m_nMoney=100;//更错误,在派⽣类中都不可以直接使⽤.在类外就更不能了.class C:protected A{void SetTelNum(int nTelNum){m_nTelNum=nTelNum;}void SetAge(int nAge){m_nAge=nAge;}void SetMoney(int nMoney){m_nMoney=nMoney;//这⾥就出现错误,因为这是基类的private成员不能⽤.}};C objC;//创建C类的对象objCobjC.m_nTelNum=123456;//注意这⾥和public的区别,这⾥错误,m_nTelNum变成了C类的protected成员objC.m_nAge=30;//错误.protected继承中基类的protected在派⽣类中是protected,这与public同相objC.m_nMoney=100;//更错误,在派⽣类中都不可以直接使⽤.在类外就更不能了.class D:private A{void SetTelNum(int nTelNum){m_nTelNum=nTelNum;}void SetAge(int nAge){m_nAge=nAge;void SetMoney(int nMoney){m_nMoney=nMoney;//这⾥就出现错误,因为这是基类的private成员不能⽤.}};D objD;//创建D类的对象objDobjD.m_nTelNum=123456;//错误,m_nTelNum变成了D类的private成员objD.m_nAge=30;//错误.private继承中基类的protected在派⽣类中是privateobjD.m_nMoney=100;//更错误,在派⽣类中都不可以直接使⽤.在类外就更不能了.从例⼦来看,三种继承从派⽣类内部引⽤来看好像没有区别,只在类外引⽤时表现不同.现在还看不出public和protected继承的区别那再看⼀个例⼦.class E:public B{void SetTelNum(int nTelNum){m_nTelNum=nTelNum;//可以因为这是B的公有成员}void SetAge(int nAge){m_nAge=nAge;//可以因为这是B的保护成员,现成变成E的protected成员}void SetMoney(int nMoney){m_nMoney=nMoney;//这个肯定不可以!}};E objE;//objE.m_nTelNum=123456;//可以//其它的两个就不能⽤了.class F:public C{void SetTelNum(int nTelNum)m_nTelNum=nTelNum;//可以因为这是C的保护成员,这⾥与public继承已经有区别但还没有表现出来 }void SetAge(int nAge){m_nAge=nAge;//可以因为这是C的保护成员,现成变成E的protected成员}void SetMoney(int nMoney){m_nMoney=nMoney;//这个肯定不可以!}};F objF;objF.m_nTel=123456;//错误,因为这是F的保护成员.注意与E类区别class G:public D{void SetTelNum(int nTelNum){m_nTelNum=nTelNum;//不可以因为这是D的private成员,注意这⾥区别}void SetAge(int nAge){m_nAge=nAge;//不可以因为这是D的private成员,注意区别}void SetMoney(int nMoney){m_nMoney=nMoney;//这个肯定不可以!}};//那G在类外就没有了可以引⽤的继承过来成员了!//这些继承⽅式是很难理解的.最好的办法就是多写代码去试.。
《基类和派生类的构造顺序》同学们,今天咱们来聊聊基类和派生类的构造顺序,这可是编程里很重要的知识哟!想象一下,基类就像是一个基础的房子框架,派生类呢,就是在这个基础框架上进行装饰和扩展的部分。
当我们创建一个派生类的对象时,先构造的是基类。
为啥呢?因为基类是基础呀,得先把基础打好。
比如说,基类里可能有一些最基本的设置,像房子的地基。
举个例子,假如基类是“动物”,派生类是“猫”。
在创建一只猫的时候,得先有动物的一些基本特征,比如有生命、能活动,然后才有猫的独特特点,像会抓老鼠、爱撒娇。
所以呀,记住这个顺序,先基类后派生类,这样我们在编程的时候就能更清楚啦!《基类和派生类的构造顺序》同学们,咱们来深入了解一下基类和派生类的构造顺序。
先把基类想象成一个大蛋糕的底层,派生类就是在这个底层上面加的各种美味装饰。
当我们要做出一个完整的蛋糕,也就是创建一个派生类对象时,肯定要先有底层的蛋糕坯,这就是基类的构造。
比如说,基类是“交通工具”,派生类是“汽车”。
那得先有交通工具的通用属性,像能移动,然后才有汽车的特别之处,比如有四个轮子、靠汽油驱动。
这个构造顺序很重要哦,如果顺序不对,就像做蛋糕先放装饰再做底层,那可就乱套啦!《基类和派生类的构造顺序》同学们,今天讲讲基类和派生类的构造顺序,这个可有意思啦!把基类当成是一棵大树的树干,派生类就是树干上长出来的树枝和树叶。
当这棵树要长大的时候,肯定得先有树干呀,这就是基类的构造。
比如基类是“水果”,派生类是“苹果”。
先要有水果的共性,像能吃、有营养,然后才有苹果的特点,比如红红的、甜甜的。
要是顺序反了,就像树先长叶子再长树干,那多奇怪呀!所以一定要记住,先构造基类,再构造派生类哟!。
5.2 基类和派生类5.2基类和派生类在C++中,当一种类被其它类继承时,被继承的类称为基类(baseclass)。
继承其它类特性的类称为派生类(derivedclass)。
从本质上看,基类是含有一种类集合中的公共特性,派生类在继承基类特性的同时能够加入自己独有的特性。
基类与派生类之间反映出下述三种不同的现象:(1)派生类是基类的具体化。
即模拟概念层次,表达“is-a”的关系。
(2)派生类是基类的延迟定义。
能够定义一种抽象基类,定义某些操作,使它们服从一定的合同,但许多可能并未实现,然后定义非抽象的派类,实现抽象基类中定义的行为。
这时派生类不是基类的具体化,而是抽象类的实现。
在JAVA中,有专门的纯虚类,称为接口,其作用就是为不同的类提供一种统一的接口,同时间接实现多继承(JAVA不支持多继承)。
(3)派生类是基类的结合。
当一种派生类有多于一种的基类时,它们组合在一起形成含有全部基类行为的类型。
这时要注意,不要用继承体现聚合关系。
5.2.1基类与派生类的阐明先看一种例子。
[例5.1] 派生类的阐明EX5_1.CPP。
继承基类的派生类定义的普通形式:class derived_class_name:access_specifier base_class_name{……};其中access_specifier能够是3个核心字之一:public、private(默认值)或protected。
派生类也称为子类、导出类。
它含有下述特点:(1)可在基类所提供有基础上包含新组员;(2)可在自己类中隐藏基类的任何组员;(3)为新类重新定义基类中的函数;[例5.2] 子类的特点EX5_2.CPP。
5.2.2派生类的继承权与访问域派生类的继承权如果不能有效在加以限制,就不能按照实际状况体现求解问题的复杂性。
因此访问权限是一种很重要的问题。
(1)对于基类的私有组员,派生类及派生类的使用者无权访问。
(2)对于基类的公有组员,则按派生类的定义,分为三种状况:私有派生,继承基类的公有组员作为自己的私有组员,这些组员只能被派生类的组员函数访问。
继承与派⽣:派⽣类对基类成员的访问控制之保护继承与私有继承(转)鸡啄⽶在编程⼊门系列上⼀节中讲了,这⼀节接着讲剩下的两种继承⽅式:保护继承和私有继承。
⼀.保护继承在保护继承⽅式中,基类的公有成员和保护成员被派⽣类继承后变成派⽣类的保护成员,⽽基类的私有成员在派⽣类中不能访问。
因为基类的公有成员和保护成员在派⽣类中都成了保护成员,所以派⽣类的新增成员可以直接访问基类的公有成员和保护成员,⽽派⽣类的对象不能访问它们,上⼀讲鸡啄⽶说过,类的对象也是处于类外的,不能访问类的保护成员。
对基类的私有成员,派⽣类的新增成员函数和派⽣类对象都不能访问。
通过上述保护继承的讲解,相信⼤家对类的有更加深刻的理解了。
假设A类是基类,B类是从A类继承的派⽣类,A类中有保护成员,则对派⽣类B来说,A类中的保护成员和公有成员的访问权限是⼀样的。
⽽对A类的对象的使⽤者来说,A类中的保护成员和私有成员都⼀样不能访问。
可见类中的保护成员可以被派⽣类访问,但是不能被类的外部对象(包括该类的对象、⼀般函数、其他类等)访问。
我们可以利⽤保护成员的这个特性,在软件开发中充分考虑数据隐藏和共享的结合,很好的实现代码的复⽤性和扩展性。
鸡啄⽶举个简单的例⼦讨论下保护成员的访问属性。
class Base{protected:int x; // 基类的保护成员};int main(){Base base;base.x = 0; // 编译报错return0;}这段代码在编译的时候会报错,错误就出在通过对象base访问保护成员x时,就像上⾯讲的,对Base类的对象base的使⽤者来说,Base 类中的保护成员x和私有成员的访问特性是⼀样的,所以对象base不能访问x,这样跟使⽤私有成员⼀样通过保护成员实现了数据的隐藏。
class Base{protected:int x; // 基类的保护成员};class Child : public Base{public:void InitX();};void Child::InitX(){x = 0;}对上⾯的派⽣类Child来说,基类Base中的保护成员x和公有成员的访问权限⼀样,所以Child类的成员函数InitX可以访问Base类的保护成员x。
在C++编程语言中,“base”是一个关键字,它在对象导向的编程中有着重要的意义。
下面我们将从多个方面来详细解释在C++语言中“base”的含义。
1. 继承的基类在C++语言中,“base”通常用来表示继承关系中的基类。
在对象导向的编程中,继承是一种重要的机制,它允许一个类(称为派生类)从另一个类(称为基类)继承属性和行为。
在这种情况下,基类通常被称为“base”,而派生类则可以通过“base”类来访问基类中定义的成员和方法。
2. 类似于父类在某些情况下,“base”还可以用来表示类似于父类的意思。
当一个类从另一个类继承而来时,它可以被认为是“base”类的一个具体实例。
在这种情况下,我们可以说派生类是基类的一种特化,它继承了基类的特征,并可以在此基础上添加自己的属性和行为。
3. 抽象基类在C++语言中,我们还可以将“base”视为抽象基类的概念。
抽象基类是一种不能实例化的类,它通常用来定义接口和规范,而不是具体的实现。
派生类可以通过继承抽象基类来实现特定的行为,这样可以提高代码的可复用性和扩展性。
“base”在这种情况下可以表示抽象基类的概念。
4. 多重继承中的基类在C++语言中,还存在多重继承的情况,即一个类可以同时从多个基类中继承属性和行为。
在这种情况下,这些基类可以被称为“base”,派生类可以通过“base”类来访问各个基类中定义的成员和方法。
多重继承使得类之间的关系更加复杂,但也为程序员提供了更大的灵活性。
从多重方面来看,“base”在C++语言中都有着重要的含义。
它可以表示继承的基类、类似于父类、抽象基类以及多重继承中的基类等概念。
掌握“base”的含义对于理解和应用C++语言中的继承机制是非常重要的,它可以帮助程序员更加灵活地设计和组织类之间的关系,从而提高代码的可维护性和扩展性。
希望通过本文的解释,读者能够对“base”在C++语言中的含义有更加清晰的认识。
很抱歉,我之前的回答似乎存在误解。
C++(面向对象的程序设计)考前必背的名词解释和简答题1. 封装封装是将数据和代码捆绑到一起,避免了外界的干扰和不确定性。
例如C++中的类,它定义了该集合中每个对象所共有的属性和方法。
2. 继承继承是让某个类型的对象获得另一个类型的对象的特征。
例如:C++中子类对父类的继承,子类具有父类的特性,同时还可以拥有自己的新特性。
3. 多态多态是指不同类型的对象接收相同的消息时产生不同的行为。
多态机制使具有不同内部结构的对象可以共享相同的外部接口,通过这种方式减小代码的复杂度。
例如函数的重载。
4. 什么是this 指针?为什么要用this 指针?this 指针是类中的一个特殊指针,当类实例化时,this 指针指向对象自己;而在类的声明时,指向类本身。
通过它不仅可以提升成员函数操作的效率,而且能简化运算符重载代码。
5. 叙述公有、私有、保护成员在类中、类外和对象中的访问权限。
类中的关键字public, private, protected 声明了类中的成员与类外之间的关系,称为访问权限。
对于public 成员来说,他们是公有的,可以在类外和对象中访问。
对于private 成员来说,他们是私有的,不能在类外和对象中访问,数据成员只能由类中的函数使用,成员函数只允许在类中调用。
对于protected 成员来说,他们是受保护的,具有半公开性质,可以在类中与子类中访问。
6. 构造函数和析构函数的作用是什么?构造函数的功能是在创建对象时,给数据成员赋初值,即给对象初始化。
析构函数的功能是释放一个对象,在对象删除前,用来做一些内存释放等清理工作。
7. 什么是类的继承和派生?继承是指一个事物可以继承其父辈全部或部分的特性,同时本身还有自己的特性。
当一个新类从一个已定义的类中派生后,新类不仅继承了原有类的属性和方法,并且还拥有自己新的属性和方法,称为类的继承和派生。
8. 派生类public 继承方式有那些特点?(1)在派生类中,基类的公有成员、保护成员和私有成员的访问属性保持不变。
派生类基类构造传参派生类基类构造传参,是面向对象编程中的一个重要概念,指的是在派生类的构造函数中通过传递参数给基类的构造函数来初始化基类的成员变量。
这个过程是实现继承关系中的一部分,能够帮助我们更好地管理和组织代码。
在理解派生类基类构造传参之前,我们首先需要了解什么是派生类和基类。
在面向对象编程中,派生类是继承了基类的属性和方法的类,也被称为子类。
基类是派生类继承属性和方法的类,也被称为父类或超类。
通过继承,派生类可以拥有基类的特征和行为,同时还可以定义自己的独特属性和方法。
在实际应用中,我们可能会遇到需要在派生类的构造函数中给基类传递参数的情况。
这时,我们可以通过派生类的构造函数使用初始化列表的方式将参数传递给基类的构造函数。
初始化列表是在派生类构造函数的括号之后,冒号之前的部分,用于初始化派生类中的成员变量和基类的构造函数。
通过派生类基类构造传参,我们可以灵活地对基类的成员变量进行初始化。
这样做的好处在于可以将一些公共属性或行为放在基类中,减少代码重复,提高代码的可维护性和可读性。
派生类基类构造传参还可以帮助我们在派生类构造函数中对基类进行个性化的初始化,以满足特定需求。
下面,我将按照从简到繁、由浅入深的方式,逐步展开对派生类基类构造传参的讨论。
一、派生类基类构造传参的基本语法和用法1. 在派生类构造函数的初始化列表中,通过基类名称和参数列表调用基类的构造函数。
2. 基类的构造函数会根据传递的参数进行相应的初始化操作。
3. 派生类在自身构造函数的初始化列表中,可以对派生类的成员变量进行初始化,也可以对基类的成员变量进行初始化。
二、为什么需要派生类基类构造传参1. 派生类可能继承了基类的成员变量,但是这些成员变量可能需要根据派生类的特定需求进行个性化的初始化。
2. 基类可能有多个构造函数,通过派生类基类构造传参可以选择合适的构造函数来初始化基类。
3. 基类构造函数可能进行一些特殊的初始化操作,通过派生类基类构造传参可以保证这些操作在派生类中得到正确的执行。
C++派⽣类与基类构造函数的调⽤顺序问题以下为转载感谢原作者关于派⽣类构造函数与基类构造函数的调⽤顺序问题,我们先看⼀下书上的说法:《⾯向对象程序设计基础(第⼆版》李师贤等,第254页:C++语⾔的基本规则是:创建⼀个派⽣类的对象时,如果基类带有构造函数,则先调⽤基类的构造函数,然后才调⽤派⽣类的构造函数。
《Thinking in C++》,刘宗⽥等译,第261页:可以看出,构造在类层次的最根处开始,⽽在每⼀层,⾸先调⽤基类构造函数,然后调⽤成员对象构造函数。
《C++ Primer Plus(第四版)中⽂版》,孙建春等译,第399页:记住:创建派⽣类对象时,程序⾸先调⽤基类构造函数,然后再调⽤派⽣类构造函数。
真的是这样吗?⼀个类的对象在实例化时,这个类的构造函数会被调⽤。
如果承认这⼀点,就会发现上述论断的⽭盾之处。
⼀个派⽣类的对象,在实例化时,不调⽤作为产⽣它的类的构造函数,⽽先去调⽤别的类的构造函数,这符合逻辑吗?再考虑⼀下基数构造函数有参数的的时候,派⽣类构造函数的定义形式,“派⽣类构造函数可以使⽤初始化列表机制将值传递给基类构造函数”(《C++ Primer Plus(第四版)中⽂版》第399页)。
如果是基类的构造函数先被调⽤,那么它所使⽤的参数从何⽽来?前两本书在说明这⼀规则时,毫⽆例外地在派⽣类构造函数和基类构造函数中使⽤cout输出⼀些信息来表明相应的构造函数被调⽤了,并以此说明构造函数的调⽤顺序。
在这⾥,我要指出的是:这⼀顺序,仅仅是这些cout输出的顺序,并不能说明是函数调⽤的顺序。
真正调⽤的过程,单纯依赖于C++是看不到的。
我们可以⽤这样的实验来证明这⼀点。
选择前两本书关于这⼀规则的任何⼀个实例,在Visual Studio中,分别对派⽣类和基类的构造函数下断点,注意:断点要下在函数定义函数名处,这样才是真正函数执⾏的起点,⽽不能下在cout语句上,那是函数体,不能说明问题。
然后调试这个程序,你会发现派⽣类构造函数的断点先中断,基类的构造函数断点后中断。
C++ 基类和派生类
9/4/2001 8:37:5· ·--··
本讲讨论基类和派生类的基本概念。
通过继承机制,可以利用已有的数据类型来定义新的数据类型。
所定义的新的数据类型不仅拥有新定义的成员,而且还同时拥有旧的成员。
我们称已存在的用来派生新类的类为基类,又称为父类。
由已存在的类派生出的新类称为派生类,又称为子类。
在C++语言中,一个派生类可以从一个基类派生,也可以从多个基类派生。
从一个基类派生的继承称为单继承;从多个基类派生的继承称为多继承。
派生类的定义格式
单继承的定义格式如下:
class <派生类名>:<继承方式><基类名>
{
<派生类新定义成员>
};
其中,<派生类名>是新定义的一个类的名字,它是从<基类名>中派生的,并且按指定的<继承方式>派生的。
<继承方式>常使用如下三种关键字给予表示:
public 表示公有基类;
private 表示私有基类;
protected 表示保护基类;
多继承的定义格式如下:
class <派生类名>:<继承方式1><基类名1>,<继承方式2><基类名2>,…
{
<派生类新定义成员>
};
可见,多继承与单继承的区别从定义格式上看,主要是多继承的基类多于一个。
派生类的三种继承方式
公有继承(public)、私有继承(private)、保护继承(protected)是常用的三种继承方式。
1. 公有继承(public)
公有继承的特点是基类的公有成员和保护成员作为派生类的成员时,它们都保持原有的状态,而基类的私有成员仍然是私有的。
2. 私有继承(private)
私有继承的特点是基类的公有成员和保护成员都作为派生类的私有成员,并且不能被这个派生类的子类所访问。
3. 保护继承(protected)
保护继承的特点是基类的所有公有成员和保护成员都成为派生类的保护成员,并且只能被它的派生类成员函数或友元访问,基类的私有成员仍然是私有的。
下面列出三种不同的继承方式的基类特性和派生类特性。
不同继承方式的基类和派生类特性
继承方式基类特性派生类特性
公有继承public public
protected
private protected
不可访问
私有继承public private
protected
private private
不可访问
保护继承public protected
protected
private protected
不可访问
为了进一步理解三种不同的继承方式在其成员的可见性方面的区别,下面从三种不同角度进行讨论。
对于公有继承方式:
(1) 基类成员对其对象的可见性:
公有成员可见,其他不可见。
这里保护成员同于私有成员。
(2) 基类成员对派生类的可见性:
公有成员和保护成员可见,而私有成员不可见。
这里保护成员同于公有成员。
(3) 基类成员对派生类对象的可见性:
公有成员可见,其他成员不可见。
所以,在公有继承时,派生类的对象可以访问基类中的公有成员;派生类的成员函数可以访问基类中的公有成员和保护成员。
这里,一定要区分清楚派生类的对象和派生类中的成员函数对基类的访问是不同的。
对于私有继承方式:
(1) 基类成员对其对象的可见性:
公有成员可见,其他成员不可见。
(2) 基类成员对派生类的可见性:
公有成员和保护成员是可见的,而私有成员是不可见的。
(3) 基类成员对派生类对象的可见性:
所有成员都是不可见的。
所以,在私有继承时,基类的成员只能由直接派生类访问,而无法再往下继承。
对于保护继承方式:
这种继承方式与私有继承方式的情况相同。
两者的区别仅在于对派生类的成员而言,对基类成员有不同的可见性。
上述所说的可见性也就是可访问性。
关于可访问性还有另的一种说法。
这种规则中,称派生类的对象对基类访问为水平访问,称派生类的派生类对基类的访问为垂直访问。
一般规则如下:
公有继承时,水平访问和垂直访问对基类中的公有成员不受限制;
私有继承时,水平访问和垂直访问对基类中的公有成员也不能访问;
保护继承时,对于垂直访问同于公有继承,对于水平访问同于私有继承。
对于基类中的私有成员,只能被基类中的成员函数和友元函数所访问,不能被其他的函数访问。
基类与派生类的关系
任何一个类都可以派生出一个新类,派生类也可以再派生出新类,因此,基类和派生类是相对而言的。
基类与派生类之间的关系可以有如下几种描述:
1. 派生类是基类的具体化
类的层次通常反映了客观世界中某种真实的模型。
在这种情况下,不难看出:基类是对若干个派生类的抽象,而派生类是基类的具体化。
基类抽取了它的派生类的公共特征,而派生类通过增加行为将抽象类变为某种有用的类型。
2. 派生类是基类定义的延续
先定义一个抽象基类,该基类中有些操作并未实现。
然后定义非抽象的派生类,实现抽象基类中定义的操作。
例如,虚函数就属此类情况。
这时,派生类是抽象的基类的实现,即可看成是基类定义的延续。
这也是派生类的一种常用方法。
3. 派生类是基类的组合
在多继承时,一个派生类有多于一个的基类,这时派生类将是所有基类行为的组合。
派生类将其本身与基类区别开来的方法是添加数据成员和成员函数。
因此,继承的机制将使得在创建新类时,只需说明新类与已有类的区别,从而大量原有的程序代码都可以复用,所以有人称类是“可复用的软件构件”。