C++ Primer Notes(15)

本文深入探讨面向对象程序设计的基本原则,包括数据抽象、继承与动态绑定等关键特性。介绍了虚函数、多态性及不同继承类型的作用和限制。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

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.友元关系不能传递也不能继承





资源下载链接为: https://pan.quark.cn/s/1bfadf00ae14 华为移动服务(Huawei Mobile Services,简称 HMS)是一个全面开放的移动服务生态系统,为企业和开发者提供了丰富的工具和 API,助力他们构建、营和推广应用。其中,HMS Scankit 是华为推出的一款扫描服务 SDK,支持快速集成到安卓应用中,能够提供高效且稳定的二维码和条形码扫描功能,适用于商品扫码、支付验证、信息获取等多种场景。 集成 HMS Scankit SDK 主要包括以下步骤:首先,在项目的 build.gradle 文件中添加 HMS Core 库和 Scankit 依赖;其次,在 AndroidManifest.xml 文件中添加相机访问和互联网访问权限;然后,在应用程序的 onCreate 方法中调用 HmsClient 进行初始化;接着,可以选择自定义扫描界面或使用 Scankit 提供的默认扫描界面;最后,实现 ScanCallback 接口以处理扫描成功和失败的回调。 HMS Scankit 内部集成了开源的 Zxing(Zebra Crossing)库,这是一个功能强大的条码和二维码处理库,提供了解码、生成、解析等多种功能,既可以单独使用,也可以与其他扫描框架结合使用。在 HMS Scankit 中,Zxing 经过优化,以更好地适应华为设备,从而提升扫描性能。 通常,ScanKitDemoGuide 包含了集成 HMS Scankit 的示例代码,涵盖扫描界面的布局、扫描操作的启动和停止以及扫描结果的处理等内容。开发者可以参考这些代码,快速掌握在自己的应用中实现扫码功能的方法。例如,启动扫描的方法如下: 处理扫描结果的回调如下: HMS Scankit 支持所有安卓手机,但在华为设备上能够提供最佳性能和体验,因为它针对华为硬件进行了
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值