c++知识总结
第八章 运算符重载
运算符和普通函数在使用的语法上虽然有所不同,但是可以将运算符看作是一种特殊的函数:操作数是函数的参数,运算结果是函数的返回值。运算符可以像函数一样重载。
operator后跟重载的运算符,operator+就是重载“+”运算符。但是不能定义c++中没有的运算符。不能重载的运算符:作用域解析符(::)、成员选择符(.)、成员指针间接引用符(.*)、条件运算符(?:)、逻辑与(&&)、逻辑或(‖)、逗号(,)、取地址符(&)
对于运算符函数来说,它或者是类的成员,或者至少含有一个类类型的参数,这意味着运算符函数只有在类类型的对象参与运算时才起作用,当运算符作用于内置类型的运算对象时,不会改变该运算符原来的含义。重载运算符时,参数基本都是引用。
重载运算符在类中使用:①作成员函数:运算符重载都在数据类中,操作类无重载,操作数少一个②做友元函数(全局函数):操作数不少
运算符函数只有在类类型的对象参与运算时才起作用,当运算符作用于内置类型的运算对象时,不会改变该运算符原来的含义
将运算符作用于类型正确的实参时,会引起重载运算符的调用,也可以像普通函数一样直接调用运算符函数,较少用
对于类类型的参数,如果仅仅只是读参数的值,而不改变参数,应该作为const引用来传递,普通算术运算符、关系运算符、逻辑运算符都不会改变参数,所以以const引用作为参数传递方式。当运算符函数是类的成员函数时,就将其定义为const成员函数。
重载输入输出
友元声明:
friend istream& operator>>(istream&is,类名&变量名);
friend ostream&operator<<(ostream&os,类名&变量名);
类外定义
istream &operator>>(istream&is,类名&变量名)
{
is>>…
teturn is;
}
ostream&operator<<(ostream&os,类名&变量名)
{
os<<…
return os;
}
第九章 组合与继承
面向对象方法的核心概念是封装、继承和多态性。c++语言中提供了数据抽象、继承和动态绑定机制支持面向对象的程序设计。以类和对象作为程序的基本构造块,利用各种关系组织类和对象,可以解决更复杂的问题。组合和继承是复用己类的两种重要途径,通过这两种途径都可以在已有类的基础上建立新类。
将一个类的对象作为另一个类的成员,被称作组合或包含。可以实现“has-a”或
“is-part-of”关系,即UML中的组合关系。
将已有类的对象作为成员,可以通过成员对象使用已有类的功能,复用其实现。类的对象成员与内置类型数据成员的声明语法相同,也可以为对象成员设置访问控制权限。
如果将嵌入的对象作为新类的公有成员,那么除了使用新类接口中提供的功能之外,还可以将其中包含的成员对象发送消息。
更常用的方式是将嵌入对象作为新类的私有成员,这时他是新类内容实现的一部分。新类中的方法可以使用成员对象提供的功能,但新类只向外部展现自己的接口,隐藏了包含的成员对象。
我们之前做的几次作业中Time类都可以作为管理类中的成员,Book类和User类也是。
继承:
1.定义:一个对象沿用另一个对象的属性和方法,并且重新赋予新的实现或者添加自己特有的属性或者方法。
2.作用:继承实现了代码重用,减少了后期的维护工作
3.内存空间:(1)父类将能够被继承的变量直接复制一份给子类,形成属于子类独立的内存空间,两个内存空间互不干扰。(2)但是子类和父类的成员函数公用同样的内存空间。
总结为:属性独立,方法共享
4.继承方式:(1)对于private继承和protected继承,所有继承的属性均为子类对象不可访问,只能通过子类公有函数访问(2)对于public继承,才是真正意义上的继承(3)所有父类私有属性全部被舍弃,但是可以通过双重函数访问:第一重为子类公有函数,第二重为父类非私有函数,将第二重丢到第一重函数里面即可
注意:对于属性而言,想被继承的就定义为公有,不想的则定义为私有,保护性一般不用;对于继承而言,一般都是使用public继承,因为这才是真正意义上的继承
5.对于子类中定义的和父类同名的函数
(1)使用子类对象调用,则优先调用子类的同名函数(此为同名覆盖);
(2)如果用父类指针指向子类对象,此指针调用的同名函数则为父类同名函数,因为父类指针指向的仅仅为子类中的父类部分。
6.构造函数:创建子类的对象时,先按继承顺序创建各个父类对象(按顺序调用父类构造函数),再创建子类对象(再调用子类构造函数)
析构函数:调用顺序和调用构造函数的顺序刚好完全相反,先子类,再后面的父类,再前面的父类(class c: publicA, public B{};则A为前面的父类,B为后面的父类)
7.多重继承:如果两个父类中有同名属性或者函数,则子类对象无法识别调用哪个,所以必须指明属性或者函数的类别(使用域名标识符::)
我们作业中对关于用户类和管理员类中的相同的操作诸如查询操作就可以单独把查询操作构建一个基类在用户和操作类中继承。
第九章 虚函数和多态
虚函数是基类的成员函数,多态性是由继承引起的。
基类在成员函数声明语句之前加上virtual关键字使得该函数执行动态绑定。语法形式为: virtual 返回类型 成员函数名(参数表);
动态绑定只对虚函数起作用,并且只有在使用含有虚函数的基类地址(指针或引用)调用时发生。虚函数在基类中用virtual关键字声明,virtual只能在类内的成员函数声明前,不能用于类外的函数定义。除了构造函数之外,任何非静态成员函数都可以是虚函数。派生类中可以重新定义基类的虚函数,这称为覆盖。在派生类中覆盖基类虚函数要求返回类型、函数名、参数个数、参数顺序都一样才行。
使用虚函数实现运行时的多态性的关键在于:必须通过基类指针访问这些函数。一旦一个函数定义为虚函数,无论它传下去多少层,一直保持为虚函数。
纯虚函数:是定义在基类中的一种只给出函数原型,而没有任何与该基类有关的定义的函数。纯虚函数使得任何派生类都必须定义自己的函数版本。否则编译报错。纯虚函数定义的一般形式:Virtual type func_name(参数列表)=0;
含有纯虚函数的基类称为抽象基类。抽象基类又一个重要特性:抽象类不能建立对象。但是抽象基类可以有指向自己的指针,以支持运行时的多态性。
有了虚函数后,用基类的指针指向派生类,调用派生类中覆盖的基类虚函数的函数,得到的结果,不再是基类中虚函数的结果,而是派生类中此函数的结果,这也被称为多态。例如我们做程序时登陆管理员与用户的登陆端,可用多态来区分。
虚函数实现多态的步骤:
在基类中讲需要多态调用的成员函数声明为virtual
在派生类中覆盖基类的虚函数,实现各自需要的功能
用基类的指针或引用指向派生类对象,通过基类指针或引用调用虚函数。