OOP概述
面向对象程序设计的核心思想是数据抽象、继承和动态绑定,使用数据抽象可以将类的接口与实现分离,使用继承, 可以定 义相似的类型并对其相似关系建模,使用动态绑定,可以在一定程度上忽略相似类型的区别,而以统一的方式使用它们的对象。
继承
通过继承联系在一起的类构成一种层次关系,通常在层次关系的根部有一个基类,其他类则直接或间接地从基类继承而来,这些继承得到类成为派生类。
在C++语言中,基类将类型相关的函数与派生类不做改变直接继承的函数区别对待。对于某些函数,基类希望他的派生类各自定义适合自身的版本,此时基类就将这些函数声明为虚函数
例:
class Quote
{
public:
std::string isbn() cosnt;
virtual double net_price(std::size_t n) const;
};
注:1、非静态成员函数后面加const(加到非成员函数或静态成员后面会产生编译错误)
2、表示成员函数隐含传入的this指针为const指针,决定了在该成员函数中, 任意修改它所在的类的成员的操作都是不允许的(因为隐含了对this指针的const引用);
3、唯一的例外是对于mutable修饰的成员。 加了const的成员函数可以被非const对象和const对象调用,但不加const的成员函数只能被非const对象调用。
派生类必须通过使用派生类列表明确指出它是从哪个基类继承而来的。派生列表的形式是:首先一个冒号,后面紧跟以逗号分隔的基类列表,其中每个基类前面可以有访问说明符,
例:
class Bulk_quote : public Quote //Bulk_quote继承了Quote
{
public:
double net_price(std::size_t) const override;
};
因为在Bulk_quote在它的派生列表中使用了public关键字,因此我们完全可以把Bulk_quote的对象当初Quote的对象来使用。
派生类必须在其内部对所有重新定义的虚函数进行声明。
动态绑定
通过使用动态绑定,我们能用同一段代码分别处理Quote和Bulk_quote的对象。
例:
double print_total(ostrem &os,const quote&item,size_t n)
{
//根据传入item形参的对象类型调用Quoe::net_price
//或者Bulk_quote::net_price
double ret = item.net_price(n);
os << “ISBN:” << item.isbn() << " #sold:" << n << “total due” << ret <<endl;
return ret;
}
因为函数print_total的item形参是基类Quote的一个引用,我们既能使用基类的Quote的对象调用该函数,也能使用派生类的Bulk_quote的对象调用它,又因为print_total是使用引用类型调用net_price函数的,实际传入print_toatl函数的对象类型将决定到底执行哪个版本的net_price。
在C++中,我们使用基类的引用(或指针)调用一个虚函数时将发生动态绑定。