1、virtual ClxTest();void DoSomething();lxTest.cpp文件内容:#include lxTest.h#include using namespace std;ClxTest:ClxTest()ClxTest()void ClxTest:DoSomething()cout Do something in class ClxTest! DoSomething();通过上面的方法就实现了类ClxExp的接口与实现的分离。请注意两个文件中的注释。类ClxExp里面声明的只是接口而已,而真正的实现细节被隐藏到了类ClxImplement里面。为了能在类ClxExp中
2、使用类ClxImplement而不include头文件lxImplement.h,就必须有前置声明class ClxImplement,而且只能使用指向类ClxImplement对象的指针,否则就不能通过编译。在发布库文件的时候,我们只需给用户提供一个头文件lxExp.h就行了,不会暴露类ClxExp的任何实现细节。而且我们对类ClxTest的任何改动,都不需要再给用户更新头文件(当然,库文件是要更新的,但是这种情况下用户也不用重新编译!)。这样做还有一个好处就是,可以在分析阶段由系统分析员或者高级程序员来先把类的接口定义好,甚至可以把接口代码写好(例如上面修改后的lxExp.h文件和lxEx
3、p.cpp文件),而把类的具体实现交给其他程序员开发。上文还没有考虑到类与类之间的继承关系。下面我们就来具体的谈谈这个方面。还是以上面提到的那篇文章中的例子来说明。执行类:接口类:lxExp.h文件内容:lxExp.cpp文件内容:但是,如果类ClxExp是另一个类的子类,而在类ClxExp中要调用基类的方法,那上面的方案就不行了。比如说,类ClxExp的基类是下面的样子:class ClxInFClxInF();virtual ClxInF();bool InitSet();virtual void DoSomething();相应的类ClxExp的声明变成了如下的形式:class ClxE
4、xp : public ClxInF现在,假设我们必须在类ClxExp的DoSomething()方法中根据InitSet()的返回值来确定是否执行操作。最简单的实现方法是把类ClxExp的DoSomething()方法改成下面的样子:if (InitSet()可是如果这样的话,接口与实现就没有彻底的分离,因为实现细节被暴露到了接口类中。为了避免这种情况发生,我们就必须把对基类ClxInF的方法InitSet()调用放到执行类ClxImplement当中。可是怎么在执行类ClxImplement当中调用接口类ClxExp的基类ClxInF的方法呢?其实很简单,因为类ClxExp是类ClxInF
5、的子类,那么它也就继承了类ClxInF的方法,只要把类ClxExp的this指针传给类ClxImplement,就可以通过这个指针来调用类ClxExp的方法,当然也可以调用类ClxExp从基类ClxInF继承来的方法。下面是修改后的代码:lxImplement.h文件内容:/ 包含声明类ClxExp的头文件/ 构造函数,传入类的ClxExp的指针ClxImplement(ClxExp *plxExp);/ 定义一个类ClxExp的指针,可以通过该指针调用类ClxExp从基类继承下来的方法ClxExp *m_plxExp;ClxImplement(ClxExp *plxExp)m_plxExp
6、= plxExp;if (m_plxExp-InitSet()对于类ClxExp来说,只要修改一下它的构造函数就行了,其他都不用修改。m_pImpl = new ClxImplement(this);这样,我们就解决了前面所提到的问题。当然,也许有人会说,让类ClxImplement也从类ClxInF继承不是更简单吗?那样就可以在类ClxImplement中直接调用类ClxInF的方法,也不用添加什么代码。可是我们知道公有继承是的子类与基类是IS-A的关系。也就是说子类是一种基类,就像说轿车是一种汽车一样。可是,在我们例子中,类ClxImplement只是类ClxExp的一个执行类而已,跟类ClxExp的基类ClxInF没有一点儿关系,更不要说是一种ClxInF了。所以不能让类ClxImplement从类ClxInF继承。
copyright@ 2008-2022 冰豆网网站版权所有
经营许可证编号:鄂ICP备2022015515号-1