业务与软件C++语言项目C进阶.pdf
![资源得分’ title=](/images/score_1.gif)
![资源得分’ title=](/images/score_1.gif)
![资源得分’ title=](/images/score_1.gif)
![资源得分’ title=](/images/score_1.gif)
![资源得分’ title=](/images/score_05.gif)
《业务与软件C++语言项目C进阶.pdf》由会员分享,可在线阅读,更多相关《业务与软件C++语言项目C进阶.pdf(36页珍藏版)》请在淘文阁 - 分享文档赚钱的网站上搜索。
1、1业务与软件C+语言项目C+进 阶录第 一 章 类、接口.71.1 Handle-Body与接口、抽象接口.71.2 多继承、与菱形缺陷、th is跳转等.131.3 C+多态的两种多态形式和区别.18第 二 章 重 载.182.1 函数重载.192.2 运算符重载.20第 三 章 模 板.293.1 模块函数.293.2 模块类.313.3 STL标准模板库.34附录:参考资料.39ISC基本知识附录:名词解释前S我们在C+基础课程中已经了解了C+的一些基本概念,知道了什么是类什么是对象。也了解了继承、封装、多态等C+面向对象的基本特征,本课程主要是更进一步探讨一下C+一 些 基本模型的应用
2、,加深对概念的理解,由于课程时间有限,C+,模型和内容又如此之多,对任何一个模型都无法深入进去,所以只能泛泛而谈。第一章 类、接口学习要求:1、了解类的继承、封装等概念之间的关系2、了解什么是接口,什 么是虚函数,它有什么样的特点。学会使用接口编程的思想本章节主要介绍C+中的类、接口。类,包涵了一组数据和一组基于数据上的一组方法。它描述了一个对象的属性、状态和行为;接口,它只是描述了一个对象的简单的行为。有关类的基本概念:Class namesClass membersMember FunctionsStatic Member FunctionsUnionsC+Bit FieldsNested
3、 Class DeclarationsType Names in Class ScopeMultiple Base ClassesVirtual FunctionsAbstract ClassesControlling Access to Class Membersprivate Members31SC基本知识 附录:名词解释protected Memberspublic MembersAccess Specifiers for Base Classes,priavte,public protectedFriendsConstructorsDestructorsConversion Funct
4、ionsthe new operator and the delete operatorCopying Constructor FunctionsInterface1.1 Handle-Body与接口、抽象接口在C+中封装的概念是把一个对象的外观接口同实际工作方式(实现)分离开来,但是C+的封装是不完全的,编译器必须知道一个对象的所有部分的声明,以便创建和管理它。我们可以想象一种只需声明一个对象的公共接口部分的编程语言,而将私有的实现部分隐藏起来。C+在编译期间要尽可能多地做静态类型检查。这意味着尽早捕获错误,也意味着程序具有更高的效率。然而这对私有的实现部分来说带来两个影响:一是即使程序员不
5、能轻易地访问实现部分,但他可以看到它;二是造成一些不必要的重复编译。然而C+并没有将这个原则应用到二进制层次上,这是因为C+的类既是描述了一个接口同时也描述了实现的过程,示例如下:class CMyString(private:const int m_cch;char*m_psz;public:CMyString(const char*psz);-CMyStringO;int Length()const;41SC基本知识 附录:名词解释int Index(const char*psz)const;CMyStirng对外过多的暴露了内存布局实现的细节,这些信息过度的依赖于这些成员变量的大小和顺序
6、,从而导致了客户过度依赖于可执行代码之间的二进制耦合关系,这样的接口不利于跨语言跨平台的软件开发和移植。1.1.1 Handle-Body 模式解决这个问题的技术有时叫句柄类(handle classes)或 叫“Cheshire Cat”1 。有关实现的任何东西都消失了,只剩一个单一的指针“m_pThis”。该指针指向一个结构,该结构的定义与其所有的成员函数的定义一样出现在实现文件中。这样,只要接口部分不改变,头文件就不需变动。而实现部分可以按需要任意更动,完成后只要对实现文件进行重新编译,然后再连接到项目中。这里有这项技术的简单例子。头文件中只包含公共的接口和一个简单的没有完全指定的类指针
7、。class CMyStringHandle(private:class CMyString;CMyString*m_pThis;public:CMyStringHandle(const char*psz);CMyStringHandle();int Length()const;int Index(const char*psz)const;;CMyStringHandle:CMyStringHandle(const char*psz)5ISC基本知识附录:名词解释:m_pThis(new CMyString(psz);()CMyStringHandle:-CMyStringHandle()(d
8、elete m_pThis;)int CMyStringHandle:Length()(return m_pThis-Length();)int CMyStringHandle:Index(const char*psz)(return m_pThis-Index(psz);)这是所有客户程序员都能看到的。这行class CMyString;是一个没有完全指定的类型说明或类声明(一个类的定义包含类的主体)。它告诉编译器,Cheshire是一个结构的名字,但没有提供有关该结构的任何东西。这对产生-一个指向结构的指针来说已经足够了。但我们在提供一个结构的主体部分之前不能创建一个对象。在这种技术里,包
9、含具体实现的结构主体被隐藏在实现文件中。在设计模式中,这就叫做Handle-Body模式,Handle-Body只含有一个实体指针,服务的数据成员永远被封闭在服务系统中。Handle-Body模式如下:61SC基本知识附录:名词解释classHandle图1 Handle-Body模 式(句柄类做为接口)Handle-Body的布局结构永远不会随着实现类数据成员的加入或者删除或者修改而导致Handle-Body的修改,即Handle-Body协议不依赖于C+实现类的任何细节。这就有效的对用户的编译器隐藏了这些斜街,用户在使用对这项技术时候,Handle-Body接口成了它唯一的入口。然而Han
10、dle-Body模式也有自己的弱点:1、接口类必须把每一个方法调用显示的传递给实现类,这在一个只有一个构造和一个析构的类来说显然不构成负担,但是如果一个庞大的类库,它有上百上千个方法时候,光是编写这些方法传递就有可能非常冗长,这也增加了出错的可能性。2、对于关注于性能的应用每一个方法都得有两层的函数调用,嵌套的开销也不理想3、由于句柄的存在依然存在编译连接器兼容性问题。接口和实现分离的Handle-Body。1.1.2抽象接口使用了“接口与实现的分离”技术的Handle-Body解决了编译器/链接器的大部分问题,而C+面向对象编程中的抽象接口同样是运用了“接口与实现分离”的思想,而采用抽象接口
11、对于解决这类问题是一个极其完美的解决方案。71SC基本知识附录:名词解释1、抽象接口的语言描述:class IMyStringvirtual int Length()const=0;这表示是一个纯虚函数,具有纯虚函数的接口virtual int Index(const char*psz)const=0;2、抽象接口的内存结构:图 2 抽象接口的内存布局3、抽象接口的实现代码:接口:class IMyStringvirtual int Length()const=0;这表示是一个纯虚函数,具有纯虚函数的接口virtual int Index(const char*psz)const=0;):实现
12、:81SC基本知识 附录:名词解释class CMyString:public IMyString(private:const int m_cch;char*m_psz;public:CMyString(const char*psz);virtual-CMyString();int Length()const;int Index(const char*psz)const;)从上面采用抽象接口的实例来看,抽象接口解决了Handle-Body所遗留下来的全部缺陷。抽象接口的一个典型应用:抽象工厂(AbstractFactroy)图3抽象工厂模式1.2 多继承与菱形缺陷、this跳转等91SC基本知
13、识 附录:名词解释多重继承是C+语言独有的继承方式,其它几乎所有语言都秉承了单一继承的思想。这是因为多重继承致命的缺陷导致的:1.2.1 菱形缺陷当继承基类时,在派生类中就获得了基类所有的数据成员副本。假如类B从A 1和A 2两个类多重继承而来,这样B类就包含A l、A 2类的数据成员副本。考虑如果A l、A 2都从某基类派生,该基类称为B a s e,现在继承关系如下:Base图4菱形继承关系我们C+语言来描述这种继承关系:c l a s s B a s e .;c l a s s A l :p u b l i c B a s e .;c l a s s A 2 :p u b l i c B
14、 a s e .;c l a s s B :p u b l i c A l,p u b l i c A 2 .;那么A l、A 2都具有B a s e的副本。这样B就包含了B a s e的两个副本,副本发生了重叠,不但增加了存储空间,同时也引入了二义性。这就是菱形缺陷,菱形缺陷时间是两个缺陷:1、子对象重叠2、向上映射的二义性。菱 形 缺 陷 的 其 中 一 种解决办法将1 01SC基本知识 附录:名词解释在C+世界里最广泛的使用虚拟继承解决菱形缺陷的应用便是标准C+的输入/输出iostream;图5标准C+的输入/输出1.2.2多重接口与方法名冲突问题(Siamese twins)对继承而来
15、的虚函数改写很容易,但是如果是在改写一个“在两个基类都有相同原型”的虚函数情况就不那么容易了。提出问题:假设汽车最大速度的接口为IC ar,潜艇最大速度的接口为IBoat,有一个两栖类的交通工具它可以奔跑在马路上,也可以航行在大海中,那么它就同时拥有ICar、IBoat两种交通工具的最大速度特性,我们定义它的接口为ICarBoat;class IC ar(virtual int GetMaxSpeed()=0;class IBoat(virtual int GetMaxSpeed()=0;我们先对ICarBoat的接口做一个尝试:class CCarBoat111SC基本知识 附录:名词解释v
16、irtual int GetMaxSpeed();既完成ICar的GetMaxSpeed()接口 方法又完成IBoat的接口方法?显然不能够);解决问题:显然上面这个尝试根本就无法成功,只用一个实现方法,怎么能够求出这个ICarBoat交通工具奔跑在马路上的最高时速,同时也能够求出航行在大海上的最大航行速度呢。上面这一问题矛盾就在一一个方法,却需要两个答案。看来ICarBoat要返回两个答案就必须有两个方法了,我们假设一个方法是求在陆地上奔跑的速度,名称为GetCarMaxSpeed();另一个方法是求在大海上航行的最大速度,名称为GetBoatMaxSpeed();那这两个方法又怎么和Get
17、MaxSpeed()接口方法联系起来呢;幸运的是,我们找到了解决办法,而且解决办法有很多种,下面介绍一下继承法。class IXCar:public ICar(virtual int GetMaxSpeed()(GetCarMaxSpeed();virtual int GetCarMaxSpeed()=0;);class IXBoat:public IBoatvirtual int GetMaxSpeed()(GetBoatMaxSpeed();)virtual int GetBoatMaxSpeed()=0;);classCCarBoat:public IXCar,public IXBoat
18、121SC基本知识附录:名词解释virtual int GetCarMaxSpeed()virtual int GetBoatMaxSpeed()图 6 多重接口与方法名冲突问题1.2.3 this 跳转th is跳转是指的“对象同一性”问题。在单一-继承的世界内,无论继承关系怎么复杂,针对于同一对象,无论它的子类或者父类的th is指针永远相等。即如果有下面的模型:13ISC基本知识附录:名词解释图 7 B从A继承的关系图那 么 对于一个已经实例化B类的对象bObject,永远有(B*)febObject=(A*)&bObject 成立。但是在多继承的世界内,上面的等式就不能恒成立,对象的同
19、一性受到了挑战。特别的是,在多继承世界内如果图四的菱形关系存在情况下,如果对于已经实例化B类的对象bObject;(Base*)(Al*)febObject!=(Base*)(A2*)&bObject 成立,当这种事情发生的时候我们就只能特殊处理了。这种情况在COM 应用中处处都会发生。1.3 C+多态的两种多态形式和区别C+有两种多态多态形式:1、编译时刻多态,编译时刻多态依靠函数重载或者模板实现2、运行时刻多态。运行时刻多态依靠需函数虚接口实现第二章 重载学习要求:1、了解什么是函数重载,什么是运算符重载2、学会运用智能指针,仿函数141SC基本知识 附录:名词解释在C+的世界里,有两种重
20、载:函数重载和运算符重载,函数重载就采用采用参数匹配的原则,进行重载的,它是一种编译时刻的多态。而运算符重载,使采用改写或者说重新定义C+的内嵌运算符的方法。有关重载的基本概念:Overloaded FunctionsOverloaded OperatorsDeclaration MatchingArgument MatchingArgument Types MatchingArgument Counts MatchingC+Unary OperatorsBinary OperatorsSmart PointerFunction objects1.1函数重载函数重载方法是在当前范围内选择一个最
21、佳匹配的函数声明供调用该方法者使用。如果一个适合的函数被找到后,这个函数将会被调用,在 这 里“适合的”是指按下列顺序匹配的符合下面条件的:1、一个精确匹配的函数被找到2、一个参数只有细微的差别,儿乎可以忽略不计的。3、象类似通过子类向父类转化达到参数匹配的4、通过正常转化函数进行类型转换,能够达到参数匹配到的。5、通过用户自定义的转化函数(如转化运算符或者构造函数)达到参数匹配的6、参数是采用省略符号函数重载的方法基本上有:1、根据函数参数数据类型的不同进行的重载;2、根据参数个数的不同进行的重载;3、缺省参数上的重载15ISC基本知识 附录:名词解释我们在这里把缺省参数也称为一种函数重载,
22、实际上它并不是严格意义上的重载。在使用缺省参数时必须记住两条规则。第一,只有参数列表的后部参数才可是缺省的,也就是说,我们不可以在一个缺省参数后面又跟一个非缺省的参数。第二,一旦我们开始使用缺省参数,那么这个参数后面的所有参数都必须是缺省的。第三,缺省参数只能放在函数声明中。第四,缺省参数可以让声明的参数没有标识符。4、返回值重载特别注意,在C+中并没有根据返回返回值的不同进行重载的,即我们不能定义这样的函数:v o i d f ();i n t f ();在C+中这样的函数声明方法是被禁止的,但是我们有时间可能又需要这样的重载方法,我们又怎么实现呢,其实很简单,j i a n g函数的参数进
23、行扩展,将这个函数返回值的数据类型,做为扩展参数的数据类型来。如下:v o i d f (v o i d);v o i d f (i n t);此时这个例子中的参数列表的数据,只在编译时刻起到分练函数的作用,在运行时刻并不起到传值作用,模板中经常都应用到了这种方法。1.2运算符重载你可以重新定义C+绝大多数内嵌运算符的实现方法和功能,这些重定义的或者说重载的运算符,有可能全局作用的,也有可能作用在类基础之上的,运算符重载的实现可能以类的成员函数的形式出现,也有可能以全局性的函数的身份出现。在C+中重载运算符的名字为operatorX,在 这 里x是一个可重载的运算符,如:重载加法运算符,你需要
24、定义一个 名 为operator+的函数,然后实现他,其它的类似定义就可以了,例如:C l a s s c o m p l e x /v e r y s i m p l i f i e d c o m p l e x161SC基本知识附录:名词解释d o u b l e r e,i m;p u b l i c:c o m p l e x (d o u b l e r,d o u b l e i):r e(r),i m(i);c o m p l e x o p e r a t o r+(c o m p l e x);c o m p l e x o p e r a t o r*(c o m p l
25、 e x););定义了 c o m p l e x这个复数的一个简单的实现概念模型。一个复数是由一对d o u b l e类型的数据组成,并定义了这个复数的两个方法,加法运算c o m p l e x:o p e r a r t o r+()和乘法运算c o m p l e x:o p e r a t o r*().现在我们就能够实现下面的复数表达式了:v o i d f ()c o m p l e x a 二c o m p l e x (1 ,3.1);c o m p l e xc o m p l e x (1.2,2);c o m p l e xb;bca =b +c;b =b +c *a
- 配套讲稿:
如PPT文件的首页显示word图标,表示该PPT已包含配套word讲稿。双击word图标可打开word文档。
- 特殊限制:
部分文档作品中含有的国旗、国徽等图片,仅作为作品整体效果示例展示,禁止商用。设计者仅对作品中独创性部分享有著作权。
- 关 键 词:
- 业务 软件 C+ 语言 项目 进阶
![提示](https://www.taowenge.com/images/bang_tan.gif)
限制150内