C++ Primer Notes(15)

来源:互联网 发布:淘宝 国外怎么用 编辑:程序博客网 时间:2024/06/08 03:45

1.面向对象程序设计的核心思想:数据抽象、继承和动态绑定。

2.对于某些函数,基类希望派生类各自定义自己的版本,此时基类就将这些函数定义为虚函数,加virtual关键字。

3.动态绑定:当使用基类的引用或指针作为(virtual)函数的形参时,我们既能使用基类对象调用该函数,也能用派生类调用,具体执行哪个版本是根据传入函数的对象类型来决定的。(总之,动态绑定只有当我们使用基类的指针或引用调用虚函数时才会发生)

4.基类通常都应该定义一个虚析构函数,即使该函数不执行任何实际操作也是如此。

5.任何构造函数之外的非static函数都可以是虚函数,关键字virtual只能出现在类内部的声明语句。

6.虚函数执行动态绑定,普通成员函数的解析过程发生在编译时而非运行时。


7.公有继承(public)、私有继承(private)、保护继承(protected)是常用的三种继承方式。

公有继承(public)

公有继承的特点是基类的公有成员和保护成员作为派生类的成员时,它们都保持原有的状态,而基类的私有成员仍然是私有的,不能被这个派生类的子类所访问。

私有继承(private):继承来的成为自己的私有成员

私有继承的特点是基类的公有成员和保护成员都作为派生类的私有成员,并且不能被这个派生类的子类所访问。

保护继承(protected):继承来的成为自己的保护成员

保护继承的特点是基类的所有公有成员和保护成员都成为派生类的保护成员,并且只能被它的派生类成员函数或友元访问,基类的私有成员仍然是私有的。


8.派生类能访问公有成员而不能访问私有成员;protected成员:派生类的成员和友元可以访问,其它用户禁止访问,需要注意的是派生类的成员和友元只能通过派生类对象来访问基类的protected成员,派生类对于基类对象的protected成员没有任何访问特权:

class Base{

protected:

   int prot_mem;

};

class Snake:public Base {

   friend void clob(Snake&);//能访问Snake::prot_mem;

   friend void clob(Base&);//不能访问Base::prot_mem;

   int j;

};

void clob(Snake& s) {s.j=s.port_mem=0;}//正确,clob能访问Snake对象的private和protected成员;

void clob(Base& b) {b.prot_mem=0;}//错误,clob不能访问Base的protected成员


9.可以把派生类的对象当成基类对象使用;

derivedclass B;  //派生类对象

baseclass *p=&B  //基类指针绑定到派生类上,隐式类型转换

10.每个类控制自己的成员初始化过程,因此派生类必须使用基类的构造函数来初始化它的基类部分;

11.如果基类定义了一个静态成员,则在整个继承体系中只存在该成员的唯一定义。

12.一个类必须声明并且定义了之后才能用作基类。


13.静态类型:表达式或变量在编译时总是已知的,是变量声明时的类型或表达式生成的类型。

动态类型:变量或表达式表示的内存中的对象的类型,直到运行时才确定。

如果表达式既不是指针 也不是引用,则它的静态类型和动态类型永远一致。基类的指针或引用的静态类型和动态类型有可能不一致(多态性的根本)。

14.派生类向基类的自动类型转换只对指针或引用类型有效!在派生类型和基类类型之间不存在这样的转换。当我们用一个派生类对象为一个基类对象初始化或赋值时,只有该派生类对象中的基类部分被拷贝、移动或赋值,它的派生部分被忽略掉。

15.综合13、14:尽管自动类型转换只对指针或引用类型有效,但继承体系中的大多数类仍然定义了拷贝控制成员,因此,我们通常能够将一个派生类对象拷贝、移动或赋值给一个基类对象。


16.所有虚函数必须都有定义,对虚函数的调用可能在运行时才被解析。对非虚函数的调用在编译时绑定,通过对象进行的函数调用(虚或非虚)调用也在编译时绑定。

17.一个派生类的函数如果(想要)覆盖了某个继承而来的虚函数,则它的形参列表必须与被它覆盖的基类函数一致。

18.派生类如果定义了一个与基类中虚函数名字相同但参数列表不同的函数,这仍然是合法的行为,编译器会认为只是一个新的函数,这个函数并不会覆盖基类中的版本。就实际的编程习惯而言,这种声明往往意味着发生了错误,因为我们可能原本希望覆盖基类中的虚函数,但是不小心把形参列表弄错了,这种错误很难通过调试解决,c++11标准中我们可以使用override关键字说明派生类中的虚函数(加在最后面),如果我们使用了override标记了某个函数,但该函数并没有覆盖基类中的虚函数,此时编译器就会报错。

19.如果一个派生类虚函数需要调用它的基类版本,但是没有使用作用域运算符,则在运行时该调用会被解析为对派生类版本自身的调用,从而导致无限递归。


20.抽象基类,纯虚函数(=0),含有(或者未经覆盖直接继承)纯虚函数的类是抽象基类,我们不能(直接)创建一个抽象基类的对象,我们可以定义它的派生类的对象,前提是派生类覆盖了纯虚函数,否则派生类还是抽象基类。

21.派生类构造函数只能初始化它的直接基类。

22.类有三种用户:普通用户、类的实现者和派生类,普通用户使用类的对象,只能访问类的公有接口,实现者编写类的成员和友元,只有它能访问私有成员;派生类能访问公有接口和受保护成员。

23.友元关系不能传递也不能继承





原创粉丝点击