继承是面向对象的一种很重要的特性,先来复习基类的基本知识:

先上一段代码:

 # ifndef  TABLE00_H
# define TABLE00_H
# include "string";
using std::string;
class Player
{
private:
string first_name;
string last_name;
bool SEAT;
public: //注意,这里只是头文件,进行函数声明的地方
//Player(const string & fn = "none", const string & ln = "none", bool symbol = false);
Player(const string & fn , const string & ln , bool symbol);
//注意,在最开始设计类的时候,可能我们并没有注意到 要使用 &引用 和const,使用& 和const的契机是:
//1 使用& 是因为,初始化类型为 string类型或者c类型字符串,进行这种非基类数据类型复制的时候,都会耗费大量内存和时间,所以采用引用&
// 2 使用const 的契机: 因为这里采用了引用,这种做法是为了不改变被引用引用的对象。思考将引用符号去掉,是否还有加const的必要性?
void NameShow()const;
bool SeatVerify()const;
//思考:在函数名后面加const,是因为在写函数体之前就想好了函数的功能是否改变成员变量,如果函数的功能不改变成员变量,就添加const,
//说白了这是一种从顶层到底层的设计,我们明白了函数的功能不改变成员变量,所以为了防止写函数体的过程改变成员变量,我们加了一个const。
// 一般的 const加在谁的前面。就是用来修饰谁的,加在返回类型前面,就是修饰返回值,加在形参前面,即修饰形参,则加在函数名后面,是修饰函数体的,具体的也就是
//不改变类对象的成员值。即这种函数称之为常成员函数。
//思考:当函数代码比较短的时候,可否在头文件直接使用内联函数将函数体键入?
void SetSeat(bool); };
//以下为共有继承类声明部分
class RePlayer :public Player
{
private:
unsigned int ratio;
public:
RePlayer(unsigned int, const string & fn, const string & ln, bool symbol);
RePlayer(unsigned int, const Player & np);
int Ratio() const;
void InitialRatio(unsigned int);
}; # endif

先复习基本知识:

1  # ifndef TABLE00_H...# endif 表明:如果之前没有定义TABLE00_H段,则编译# ifndef TABLE00_H...# endif之间程序段,否则不编译,这 能够避免一个文件被多个重叠文件连续包含时报错,比如B头文件包含了A文件,C头文件包含了B文件和A文件,那么如果没加# ifndef TABLE00_H...# endif ,则会因为重复定义报错,因此在写头文件时,一律写上 # ifndef TABLE00_H...# endif可以避免程序的报错问题。

2  对一个类而言,构造函数是十分重要的一个环节,构造函数存在的意义是:让私有成员变量被初始化,我们应当始终注意这一初衷,只有这样,我们才能设计正确的形参。

3  我们应该注意引用&变量的使用契机,当传递的参数是复杂数据类型(比如类和c类型的字符串),由于巨大的内存开销和调度,采用引用的方式无疑是一种高效的方式

4  上述代码段17,18,35行的函数成为:常成员函数,在此,先声明函数声明结尾const的作用,使得程序体不能改变私有成员变量的值(否则报错),比如成员显示函数,可以使用常成员函数

上述代码28-37行为继承类的生命,从这个声明我们可以得到这样一些基本信息与结论:

1  继承类首先也是类,具有类的一般特性:包括私有成员、公有成员,以及构造函数。

2  观察继承类的构造函数。发现其构造函数同样服从:让私有成员变量被初始化.但继承类继承了基类,因此也要对基类的成员进行初始化,说白了,要对所有的成员进行初始化。

易错:

也许有人看了13,33,34行的代码,会发出这样的疑问:为何这里使用了引用变量却没有初始化,引用变量在定义变量时不是要进行初始化吗?

回答:我们在声明类,甚至在定义类的时候,本质工作是什么???本质工工作是:构造,构造一个数据类型,并不是在定义变量,只有我们在使用类(构造的数据类型)去定义对象的时候,我们才是真正的定义了一个变量,所以 定义类的过程,并不是定义变量的过程,所以并不必要对&进行初始化,说白了,此时的引用&只是一个空壳子,并不实际的分配内存,进行初始化这些功能。

进行了类声明之后,但成员函数还未得到定义,为此,给出类定义:

 # include "table00.h"
# include "iostream"
using std::string;
using std::cout;
using std::endl;
/*class Player //如果在函数体文件再声明class Player则会出现重定义的情况!!!,所以采用这种做法是错误的。
{
private:
string first_name;
string last_name;
bool SEAT;
public:
Player(const string & fn = "none", const string & ln = "none", bool symbol = false)
{
first_name = fn;
last_name = ln;
SEAT = symbol;
}
void NameShow()const //注意在函数体中,这个const也不能丢舎.
{
cout << first_name << "," << last_name << endl;
}
bool SeatVerify()const
{
return SEAT;
}
void SetSeat(bool change_seat)
{
SEAT = change_seat;
}
};*/
//验证上述写法和下述写法哪个更好。以及对于作用域有没有更好的表示方法。
//Player::Player(const string & fn = "none", const string & ln = "none", bool symbol = false)
Player::Player(const string & fn , const string & ln, bool symbol ) {
first_name = fn;
last_name = ln;
SEAT = symbol;
}
void Player:: NameShow()const //注意在函数体中,这个const也不能丢舎.
{
cout << first_name << "," << last_name << endl;
}
bool Player:: SeatVerify()const
{
return SEAT;
}
void Player:: SetSeat(bool change_seat)
{
SEAT = change_seat;
}
//要认识到面向对象这个词的含义:函数的作用尽管也是为了完成一个功能,但更多的是完成对数据的操作,即我们更关注数据本身
// 成员函数的本质在于:服务于成员变量(通常情况是这样),所以在进行成员函数设计的时候,我们所关注的重点是:对成员变量进行何种操作,完成何种功能
//一定要注意主体对象是成员变量。 RePlayer::RePlayer(unsigned int v, const string & fn, const string & ln, bool symbol) : Player(fn, ln, symbol)
{
ratio = v;
// first_name = fn; 注意,如果我们试图直接访问基类私有变量,是有问题的
// last_name = ln; 但我们需要在调用继承类构造函数之前,调用基类构造函数。
// SEAT = symbol;
}
//这两条都是继承类构造函数,需要在调用之前调用基类构造函数,因此需要先初始化基类构造函数。
RePlayer::RePlayer(unsigned int v, const Player & np) : Player(np)
{ //需要注意的是:如果 前面定义 unsigned int v =0;则后面的np也要赋初值
//注意,这里的写法发生了重定义。
ratio = v;
// first_name = fn; 注意,如果我们试图直接访问基类私有变量,是有问题的
// last_name = ln; 但我们需要在调用继承类构造函数之前,调用基类构造函数。
// SEAT = symbol;
}
int RePlayer:: Ratio()const
{
return ratio;
} void RePlayer::InitialRatio(unsigned int initial)
{
ratio = initial;
}

关于成员函数(也被称为接口,其实很形象!!!)有以下内容需要说明:

1  无论是基类的成员函数,还是继承类的成员函数,发现:成员函数都更侧重于:对成员变量(也称为实现,也很形象)进行了何种操作。虽然成员函数也描述了:完成了一个怎样的功能,但我们更侧重于:对成员变量完成了一种怎样的功能,也就是最终落脚点在于:成员变量发生了什么?因此,我们在写成员函数的时候,一定不能漫无目的,思考要完成一个什么功能但脱离了成员变量,一定要认识到我们的成员函数是紧紧的围绕成员变量展开的。

2 关注继承类的构造函数的实现:也就是上述,57和65行的代码。在初始化一个继承类成员(实际上包含了基类成员在内的所有成员)的时候,必然先初始化基类的成员变量,要调用继承类的构造函数,一定要首先调用其基类的构造函数,完成对基类成员变量先进行初始化。因此在进入继承类构造函数函数体之前,必然先要调用基类构造函数完成基类成员变量的初始化。

这也是为什么57行Player(fn, ln, symbol)与65行的 Player(np)会写在函数体{}的前面

3  我们注意:60行和69行的代码,当我们试图去直接访问基类私有成员变量时,程序是禁止的,也就是说,我们只能通过基类的公有函数才能访问基类的私有成员。这一点保证了父类和子类的独立性关系。

最终,我们给出函数的调用:

 # include "table00.h"
# include "iostream"
using namespace std;
int main()
{
Player player1("jack", "cracy", true);
player1.NameShow();
Player player2(player1);
player2.NameShow();
RePlayer player3(, "robert", "lin", true);
player3.NameShow();
RePlayer player4(,player2);
player4.NameShow();
system("pause");
return ; }

从代码中,可以看到:继承类可以调用基类的公有函数。

上说代码体现了类的基本思想和类继承的基本思想,下面我们给出一个更深入的探讨,来探讨基类和继承类的一些关系:

 # include "table00.h"
# include "iostream"
using namespace std;
void Show(const Player & );
int main()
{
Player player1("jack", "cracy", true);
player1.NameShow();
Player player2(player1);
player2.NameShow();
RePlayer player3(, "robert", "lin", true);
player3.NameShow();
RePlayer player4(,player2);
player4.NameShow();
Player & p = player3; //我们可以用子类去初始化父类,这是没有问题的,因为子类继承了父类的特性(成员)
p.NameShow();
Player* q= & player4;
q->NameShow(); //注意指针的访问方式和引用访问方式的区别,
//引用就相当于是别名,所以引用名就和对象名是等价的,因此可以用.访问,而指针并不等价于别名
//指针的访问要采用->。
Player player("ma", "jack", false);
//RePlayer & rt = player;//我们不可以用父类来初始化(或者赋值)子类,因为子类具有父类不具备的一些特性(子类新3定义成员),
//RePlayer * pt = &player;
Player player5("li", "zhou", false);
Show(player5);//将基类对象作为实参 传递给 基类引用,可行不报错
RePlayer player6(, "lin", "wu", true);
Show(player6);//将继承类对象作为实参 传递给 基类引用,,可行不报错
player6=player5;//试图用基类对象 赋值 继承类 对象,报错!!!
player5 = player6;//用继承类对象 赋值 基类 对象,不报错,实际上,这里使用了运算符重载!!!player& operator=(const player & )const;
Player player7(player5); //用基类对象初始化另一个基类对象,可行
Player player8(player6);//用 继承类 对象初始化 基类对象,可行。
RePlayer player9(player6);// 用 继承类 对象 初始化 继承类对象 ,可行!
RePlayer player9(player5);//用 基类对象 初始化 继承类对象,不可行!!!
system("pause");
return ; } void Show(const Player& rt)
{
rt.NameShow();
}

上述代码体现了子类和父类这样的一些特性:

当进行类似于赋值操作的时候,子类可以对父类进行赋值,因为子类继承了父类的全部特性。但不能用父类对子类赋值,因为子类有的特性,父类不一定有。

同时,注意15,17行代码的给出了引用和指针的一些区别:引用可以看做是别名,但指针并不能看成别名。因此在访问的时候是有一定区别的。

最新文章

  1. android 在 ListView 的 item 中插入 GridView 仿微信朋友圈图片显示。
  2. 【iCore3 双核心板_ uC/OS-III】例程六:信号量——共享资源
  3. Python 列表如何获得一个指定元素所在的下标
  4. chrome内核浏览器缓存资源找回方法
  5. hdu 3595 GG and MM 博弈论
  6. Eclipse选择rt.jar的源代码的位置
  7. 委托、 Lambda表达式和事件——委托
  8. Jetty9开发(1)
  9. WM_PAINT消息在窗口重绘的时候产生,那什么时候窗口会重绘(异步工作方式效率高、灵活性强,还有UpdateWindow和RedrawWindow帮忙)
  10. SpringBoot(二)Web整合开发
  11. 学习资料分享:Python能做什么?
  12. [洛谷P2024/POJ1182]食物链 - 带偏移量的并查集(2)
  13. 【不定期更新】FPGA/IC岗位常见笔试面试题总结(基础知识)
  14. UUID、GUID、SID、SUSID
  15. java 软件安装
  16. 牛客网Wannafly挑战赛25A 因子 数论
  17. 如何一键式搭建微信小程序
  18. Petrozavodsk WinterTraining 2015
  19. [python爬虫] 爬取图片无法打开或已损坏的简单探讨
  20. Opengl---gluLookAt函数详解(转)

热门文章

  1. css图片垂直水平居中及放大(实现水平垂直居中的效果有哪些方法?)
  2. 关于Inception默认配置的一个坑
  3. 自动化测试基础篇--Selenium鼠标键盘事件
  4. mysql性能排查思路
  5. vim 基础命令大全
  6. liunx搭建DHCP服务器以及DHCP中继服务器
  7. [HBase_1] HBase安装与配置
  8. celery 定时任务
  9. Ubuntu下导入PySpark到Shell和Pycharm中(未整理)
  10. 设计模式のInterpreter Patern(解释器模式)----行为模式