C++11(14):面向对象程序设计

面向对象程序设计基于三个基本概念:数据抽象、继承和动态绑定。
继承和动态绑定对程序的编写有两方面影响:一是我们可以更容易地定义与其他类似但不完全相同的新类:二是在使用这些彼此相类似的类编写程序时,我们可以在一定程度上忽略掉他们的区别。
使用数据抽象,我们可以将类的接口和实现分离,使用继承,可以定义相类似的类型并对其相类似的关系建模,使用动态绑定,可以再一定程度上忽略相似类型的区别,而以统一的方式使用它们。
派生类可以再需要重新定义的函数之前加上virtual,但不是必须的。
在c++中,当我们使用基类的引用(或指针)调用一个虚函数时将发生动态绑定。
class Quote{
public:
    Quote( ) = default;
    Quote(const std::string &book, double sales_price) : bookNo(book), price(sales_price) { }
    std::string isbn( ) const{ return bookNo; }
    virtual double net_price(std::size_t n) const
        { return n*price;}
    virtual ~Quote( ) = default;
privete :
    std::string bookNo;
protected:
    double price = 0.0;
};
我们通常会定义一个虚析构函数,即使该函数不指向任何实际操作也是如此。


基类有两种成员函数:一种是基类希望其派生类进行覆盖(override)的函数;另一种是基类希望派生类直接继承而不要改变的函数。前者是虚函数。当我们用指针或引用调用时,该调用将被动态绑定。
任何构造函数之外的非静态函数都可以是虚函数。
virtual只能出现在类内部的声明语句之前而不能用于类外部的函数定义。如果我们把以个函数声明成虚函数,则该函数的在派生类中隐式地也是虚函数。
成员函数如果么有被声明为虚函数,则其解析过程发生在编译时而非运行时。
派生类能访问基类的共有成员,而不能访问私有成员。受保护的成员允许派生类访问,同时禁止其他用户访问。

class Bulk_quote : public Quote {
public:
    Bulk_quote( ) = default;
    Bulk_quote(const std::string& , double , std::size_t ,double);
    double net_price(std::size_t) const override;
private:
    std::size_t min_qty = 0;
    double discount = 0.0;
};
如果一个派生类是共有的,则基类的公有成员也是派生类的组成部分。此外,我们将公有派生类型的对象绑定到基类的引用或指针上。
派生类可以再它覆盖的函数前使用virtual关键字, 但不是非得这么做。新标准,显示的注明覆盖,用override
一个派生类对象包含多个组成部分:自己定义的和继承来的一个或多个部分。在一个对象中,继承自基类的部分和派生类自定义的部分不一定是连续存储的。
因为派生类对象中含有与其基类对应的组成部分,所以我们能把派生类的对象当成基类对象来使用,而且我们也可以将基类的指针或引用绑定到派生类对象中的基类部分上。
编译器会隐式执行派生类到基类的转换。
这种隐式转换的特性意味着我们可以把派生类对象或派生类对象的引用用在需要基类引用的地方。同样的我们也可以吧派生类的指针要在需要基类指针的地方。

尽管派生类对象不直接初始化基类这些成员。但是必须使用基类的构造函数类初始化它的基类部分。首先初始化基类部分,然后按照声明顺序依次初始化派生类的成员。
Bulk_quote(const std::string& book, double p, std::size_t qty, double disc) : 
        Quote(book,p) , min_qty(qty) , discount(disc){ }
除非我们特别指出,否则派生类对象的基类部分会像数据成员一样进行默认初始化。如果想使用基类的其他构造函数就必须用参数类区别。

double Bulk_quote::net_price(size_t cnt) const
{
    if(cnt >= min_qty)
        return cnt * (1-discount) * price ; 
    else
        return cnt * price;
}
派生类可以访问基类的共有和受保护成员,派生类的作用域嵌套在基类的作用域之内。
尽管在语法上类说我们可以在派生类构造函数体内给它的共有或受保护的基类成员赋值,但是最好不要这么做。应该是用基类的接口。

如果基类定义了一个静态成员,在这个继承体系中只存在该成员的唯一定义。静态成员遵循通用的访问控制规则,如果基类中的成员是private的,则派生类无权访问它。假设某个静态成员可访问的,则我们既能通过基类使用它也能通过派生类使用它。

派生类的声明不用包含基类对象。

如果我们要将一个类用作基类,则该类必须已经定义而非仅仅声明。原因是,派生类中包含并且可以使用它从基类继承而来的成员,为了使用这些成员,派生类当然要知道他们是什么。因此该规定还有一层隐含的意思,即一个类不能派生它本身。
一个类是基类,同时它也可以是一个派生类:直接基类,间接基类。
直接基类出现在派生列表中,而间接基类由派生类通过其直接基类继承而来。
每个类都会继承直接基类中的所有成员。最终的派生类将包含它的直接基类的子对象以及每个间接基类的子对象。

有时候我们会定义这样一种类,我们不希望其他类继承它,或者不想考虑它是否适合做一个基类。c++11提供了一种防止继承的方法:在类名后跟一个关键字final:
class NoDerived final{ /*...*/};

一般,如果我们想把引用或指针绑定到一个对象上,则引用或指针的类型应与对象的类型一致,或者对象的类型含有一个可接受的const类型转换规则。存在继承关系的类是一个重要的例外:我们可以将基类的指针或引用绑定到派生对象上。
可以将基类的对象或指针保定到派生对象上有一层极为重要的含义:当使用基类的引用(或指针)时,实际上我们并不清楚该引用(或指针)所绑定对象的真实类型。该对象可能是基类对象,也可能是派生类对象。
和基类一样,智能指针类也支持派生类向基类的类型转换。

表达式的静态类型在编译时总是已知的,它是变量声明时的类型或表达式生成的类型;动态类型则是变量或表达式表示的内存中的对象的类型。动态类型直到运行时才可知。
基类的指针或引用的静态类型可能与动态类型不一致,如果表达式既不是用的引用也不是指针,则它的动态类型永远和静态类型一致。
当基类的指针或引用调用虚函数时,将发生多态性。
因为一个基类的对可能是派生类对象的一部分,也可能不是,所以不存在从基类想派生的自动转换。
编译器在编译时无法确定某个特定的转换在运行时是否安全,这是因为编译器只能通过检查指针或引用的静态类型类判断该转换是否合法。如果在基类中含有一个或多个虚函数,我们可以使用dynamic_cast请求以个类型转换,该转换的安全检查在运行时执行。同样的,如果我们已知某个基类向派生类的转换是安全的,则我们可以使用stati
  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值