寒星轩

There are innumerable stars in the sky, the smallest is me!

用户操作
[即时聊天] [发私信] [加为好友]
李星ID:starlee
207683次访问,排名341好友65人,关注者107
欢迎大家访问我的Blog。
主要是C++,设计模式,面向对象设计方面的技术文章。
starlee的文章
原创 98 篇
翻译 0 篇
转载 45 篇
评论 331 篇
李星的公告
郑重声明

        本BLOG所发表的 原创文章,作者保留一切权利。必须经过作者本人同意后方可转载,并注名作者(StarLee)和出处(CSDN Blog)。
作者Email:
coolstarlee(at)sohu.com
最近评论
陈诚:好象不一样,我这个共两个类,实现类和接口类
深夜才走在路上:实际上使用CLR封送C++类让人很受伤,在mc中有很多C++的特性不能使用,甚至STL都不可用
hfg :错了错了,当用一个基类的指针删除一个派生类的对象时,派生类的析构函数会被调用。

不管基类的析构函数是不是virtual ,派生类的析构函数还是会被调用的,区别只是在于基类的析构函数有没有被调用
Forrest Yu:Star Lee:

如果有两个以上的类需要包装,那又应该怎样做呢?
Forrest Yu:CLR 还是很强大的,
一些老的MFC项目可以先手动添加
#include <afx.h>,
其他的可能要加
#include <windows.h>,
然后再用这种方法.
文章分类
收藏
相册
友情链接
houdy的专栏
lijgame的专栏
lyrebing的专栏
禾青谷
存档
订阅我的博客
XML聚合  FeedSky
订阅到鲜果
订阅到Google
订阅到抓虾
订阅到BlogLines
订阅到Yahoo
订阅到GouGou
订阅到飞鸽
订阅到Rojo
订阅到newsgator
订阅到netvibes

原创 再论C++中接口与实现分离的技术收藏

新一篇: 多篇文章入选CSDN技术中心,发文纪念 | 旧一篇: 点击次数过万,发文纪念

    我在今年2月份写了篇《C++中接口与实现分离的技术》的文章,用一个很简单的例子说明了在C++中接口与实现分离的好处及实现方法。很荣幸,这篇文章被推荐到了CSDN的首页并被多家网站转载。
    可是当时写那篇文章的时候,没有考虑到类与类之间的继承关系。下面我就来具体的谈谈这个方面。
    还是以上面提到的那篇文章中的例子来说明。
    执行类:
    lxImplement.h文件内容:

#include "lxTest.h"

class ClxImplement  
{
public:
    ClxImplement();
    
~ClxImplement();

    
void DoSomething();

private:
    ClxTest m_lxTest;

    
void lxTest();
};

    lxImplement.cpp文件内容:

#include "lxImplement.h"

ClxImplement::ClxImplement()
{
}

ClxImplement::
~ClxImplement()
{
}

void ClxImplement::lxTest()
{
    m_lxTest.DoSomething();
}

void ClxImplement::DoSomething()
{
    lxTest();
}

    接口类:
    lxExp.h文件内容:

//  前置声明
class ClxImplement;

class ClxExp  
{
public:
    ClxExp();
    
virtual ~ClxExp();

 
void DoSomething();

private:
    
//  声明一个类ClxImplement的指针,不需要知道类ClxImplement的定义
    ClxImplement *m_pImpl;
};

  lxExp.cpp文件内容:

//  在这里包含类ClxImplement的定义头文件
#include "lxImplement.h"

ClxExp::ClxExp()
{
    m_pImpl 
= new ClxImplement;
}

ClxExp::
~ClxExp()
{
    
if (m_pImpl)
        delete m_pImpl;
}

void ClxExp::DoSomething()
{
    m_pImpl
->DoSomething();
}

    但是,如果类ClxExp是另一个类的子类,而在类ClxExp中要调用基类的方法,那上面的方案就不行了。比如说,类ClxExp的基类是下面的样子:

class ClxInF
{
public:
    ClxInF();
    
virtual ~ClxInF();

    
bool InitSet();

    
virtual void DoSomething();
};

    相应的类ClxExp的声明变成了如下的形式:

class ClxExp : public ClxInF
{
public:
    ClxExp();
    
virtual ~ClxExp();

    
void DoSomething();

private:
    ClxImplement 
*m_pImpl;
};

   现在, 假设我们必须在类ClxExp的DoSomething()方法中根据InitSet()的返回值来确定是否执行操作。最简单的实现方法是把类ClxExp的DoSomething()方法改成下面的样子:

void ClxExp::DoSomething()
{
    
if (InitSet())
        m_pImpl
->DoSomething();
}

    可是如果这样的话,接口与实现就没有彻底的分离,因为实现细节被暴露到了接口类中。为了避免这种情况发生,我们就必须把对基类ClxInF的方法InitSet()调用放到执行类ClxImplement当中。可是怎么在执行类ClxImplement当中调用接口类ClxExp的基类ClxInF的方法呢?其实很简单,因为类ClxExp是类ClxInF的子类,那么它也就继承了类ClxInF的方法,只要把类ClxExp的this指针传给类ClxImplement,就可以通过这个指针来调用类ClxExp的方法,当然也可以调用类ClxExp从基类ClxInF继承来的方法。下面是修改后的代码:
    lxImplement.h文件内容:

#include "lxTest.h"
// 包含声明类ClxExp的头文件
#include "lxExp.h"

class ClxImplement  
{
public:
    
//  构造函数,传入类的ClxExp的指针
    ClxImplement(ClxExp *plxExp);
    
~ClxImplement();

    
void DoSomething();

private:
    ClxTest m_lxTest;
    
//  定义一个类ClxExp的指针,可以通过该指针调用类ClxExp从基类继承下来的方法
    ClxExp *m_plxExp;

    
void lxTest();
};

    lxImplement.cpp文件内容:

#include "lxImplement.h"

ClxImplement::ClxImplement(ClxExp 
*plxExp)
{
    m_plxExp 
= plxExp;
}

ClxImplement::
~ClxImplement()
{
}

void ClxImplement::lxTest()
{
    m_lxTest.DoSomething();
}

void ClxImplement::DoSomething()
{
    
if (m_plxExp->InitSet())
        lxTest();
}

    对于类ClxExp来说,只要修改一下它的构造函数就行了,其他都不用修改。

ClxExp::ClxExp()
{
    m_pImpl 
= new ClxImplement(this);
}

    这样,我们就解决了前面所提到的问题。
    当然,也许有人会说,让类ClxImplement也从类ClxInF继承不是更简单吗?那样就可以在类ClxImplement中直接调用类ClxInF的方法,也不用添加什么代码。可是我们知道公有继承是的子类与基类是IS-A的关系。也就是说子类是一种基类,就像说轿车是一种汽车一样。可是,在我们例子中,类ClxImplement只是类ClxExp的一个执行类而已,跟类ClxExp的基类ClxInF没有一点儿关系,更不要说是一种ClxInF了。所以不能让类ClxImplement从类ClxInF继承。

发表于 @ 2006年07月04日 08:49:00|评论(loading...)|编辑

新一篇: 多篇文章入选CSDN技术中心,发文纪念 | 旧一篇: 点击次数过万,发文纪念

评论

#dynuaa 发表于2006-08-11 21:20:00  IP: 218.2.240.*
当然,也许有人会说,让类ClxImplement也从类ClxInF继承不是更简单吗?那样就可以在类ClxImplement中直接调用类ClxInF的方法,也不用添加什么代码。可是我们知道公有继承是的子类与基类是IS-A的关系。也就是说子类是一种基类,就像说轿车是一种汽车一样。可是,在我们例子中,类ClxImplement只是类ClxExp的一个执行类而已,跟类 ClxExp的基类ClxInF没有一点儿关系,更不要说是一种ClxInF了。所以不能让类ClxImplement从类ClxInF继承。

-----------------------------------

个人觉得用子类来实现父类的是一种常用的方法,没有必要太教条了。最终不过是一些二进制代码而已。即使从逻辑上来说,也可以大致认为,这样的实现类也是基类的一种。
#wingfiring 发表于2007-03-29 12:56:12  IP: 202.95.81.*
比如可能还有一个问题需要解决:
如何提供copy ctor和assignment?
这个问题不难解决,但是会导致很多的复杂性,会使得这个方法缺乏生命力。
发表评论  


登录
Csdn Blog version 3.1a
Copyright © 李星