『重构--改善既有代码的设计』读书笔记----Introduce Local Extension

Introduce Foreign Method一样,很多时候你不能修改编辑原始类,你需要为这些服务类增加一些额外的函数,但你没有这个权限或者入口。如果你只需要一个或者两个外加函数那么你可以放心的使用Introduce Foregin Method,但是如果你发现此时有很多外加函数需要在客户类代码中中添加,你就要小心了,因为你这么做你就会让客户类变得过分复杂,责任就会过分多,你会破坏客户类的单一职责性。这个时候你就可以建立一个新类,让他来包含这些你之前所添加的额外函数,让这个扩展类成为源类的子类或者包装类

在这里提供了两种标准对象技术--子类化(subclass)和包装(wrapping),在这两种方法下,作者统一称他们为本地扩展(Local Extension)。所谓本地扩展就是一个独立的类,但也是被扩展类的子类型,他可以提供源类的一切特性同时可以添加新特性,在任何使用源类的地方你都可以使用本地扩展取而代之。

坚持使用本地扩展可以让你坚守“函数和数据应该被统一封装”的原则。如果你一直把本该放在扩展类的代码零散地放置于其他类中,最终你会让这些客户类变得过分复杂,让其中的函数变得难以复用。

在子类和包装类之间做选择时,作者推荐优先使用子类,这也是有道理的,因为子类可以让你最少的去修改或者去添加源类本身有的特性,包装类你需要做一系列简单委托来维持原特性。但是子类化也是有比较麻烦的地方,就是它必须在对象创建期实施,如果你可以接管对象的创建期那当然没问题。但如果你想在对象创建之后再进行本地扩展那就有问题了。此外你还需要去面对别名问题,因为子类化对象通常是新创建一个子类对象,在这种情况下如果还有其他类引用了旧对象,你就同时有了两个对象保存了相同的原始数据。如果原数据是不可修改的,那还好没有什么问题,但如果是可修改的,你的问题就来了,因为一个修改不可能同时修改两个副本,如果你需要面对这个问题,那么你就应该使用包装类,因为只有使用包装你才可以让本地扩展(Local Extension)波及原对象,反之如果你不想让本地使用波及原来的数据,那么你应该优先使用子类化的方案。

  • 做法:
  • 建立一个扩展类,由你选择判断让他作为源类的子类或者包装类。
  • 在扩展类中加入“转型构造函数”,所谓“转型构造函数”就是指“接受原对象作为参数”的构造函数。如果采用子类化方案,那么转型构造函数应该调用适当的超类构造函数,如果采用包装类方案,那么转型构造函数应该将它得到的传入参数以实例变量的形式保存起来,用以接受委托对象。注意:使用转型构造函数更多是为了隐藏扩展类,让用户从使用上,特别是针对函数参数是源类的时候,可以做到转型构造。
  • 在扩展类中加入新特性。
  • 根据需要将原对象替换为扩展对象。
  • 将针对原始类增加的所有外加函数搬移到扩展类中。

例子:

class MfDateSub : public Date
{
    public:
        MfDateSub *nextDay() const...
        int dayOfYear()...
};

class MfDateWrap
{
    private:
        Date *m_date;
};

针对我们上一篇的例子,我们有这两种方式来引入本地扩展,一个就是子类化,一个就是包装类。我们先来看看子类化

class MfDateSub : public Date
{
    public:
        MfDateSub(const QString &dateString) :
            Date(dateString)
    {
    }
};

我们可以把基类构造函数的参数传入给基类的构造函数来完成扩展类的构造,现在我们需要同时引入一个转型构造函数,参数就是源类对象。

class MfDateSub : public Date
{
    public:
        MfDateSub(const QString &dateString) :
            Date(dateString)
    {
    }
        MfDateSub(Date *date) :
            Date(date->getTime())
    {
    }
};

现在我可以在扩展类中增加新特性,用Move Method将之前添加的额外函数nextDate()移动到这个扩展类

clien class...
static Date *nextDay(Date *arg)
{
    // foreign method
    return new Date(arg->getYear(), arg->getMonth(), arg->getDate() + 1);
}

class MfDateSub...
Date *nextDate()
{
    return new Date(getYear(), getMonth(), getDate() + 1);
}

我们再来看下使用包装类的过程

class MfDateWrap
{
    public:
        MfDateWrap(const QString &dateString)
        {
            m_date = new Date(dateString);
        }
    private:
        Date *m_date;
};

包装类的构造函数和之前有所不同,现在的构造函数只是很简单的执行一个委托动作。而转型构造函数则只是对它的字段赋值而已。

MfDateWrap(Date *arg)
{
    m_date = arg;
}

接下来就是一连串枯燥的工作,我们要为原始类的所有函数提供委托函数。

int getYar()
{
    return m_date->getYear();
}

int getMonth()
{
    return m_date->getMonth();
}

完成这个工作之后我们进行Move Method把额外函数引入到扩展类

class MfDateWrap...
Date *nextDate()
{
    return new Date(getYear(), getMonth(), getDate() + 1);
}

当然,使用包装类有一个问题就是,就是如何处理接受原始类作为参数的函数,比如

bool after(Date *arg):

由于我们无法改变原始类,我们只能在一个方向上做到兼容,即在包装类的after()函数可以接受包装类或者原始类对象,但原始类的after()函数只能接收原始类对象,不能接受包装类对象。

wrapper->after(date); // ok
wrapper->after(anotherWrapper); // ok
date->after(wrapper); // error

所以其实我们应该显示的给包装类增加一个额外的after函数进行覆写(子类化不需要,因为子类指针可以被基类指针的参数所接受)即

bool after(Wrapper *arg):

这样覆写的目的就是为了向用户隐藏包装类的存在,这是一个好策略,因为包装类的客户的确不应该知道包装类的存在,的确应该可以同样的对待原始类和包装类,但是我们无法完全隐藏包装类的存在,原因就是上文所说的,源类可能不能通过包装类的参数来完成函数的调用,特别是对于equals()这种满足交换率的来说更是如此。因此我们只能做出妥协,创建一个新的函数来单独完成针对两种类的比较

bool equalsDate(Date *arg);
bool equalsDate(MfDateWrap *arg);

这样你就不必检查未知对象的类型了。当然了子类化方案中没有这样的问题,只要我不覆写原函数就行了。但如果你覆写了原始类中的函数,那么当你寻找函数时就容易晕头转向,一般来说,我们不会在扩展类中覆写原始函数,只会添加新函数。

时间: 2024-08-11 00:59:13

『重构--改善既有代码的设计』读书笔记----Introduce Local Extension的相关文章

『重构--改善既有代码的设计』读书笔记----Introduce Explaning Variable

有时候你会遇到一系列复杂的表达式连续运算的时候,这个时候你可能根本招架不住如此长或者是如此复杂的长函数.这个时候你可以通过引用临时变量来储存他们的结果,将这些长函数的结果分成一个个临时变量来让函数清晰化.但在这里,我的想法与作者的想法是一样的,我会更倾向于去用Extract Method去将复杂函数弄清晰,而尽量不去Introduce Explaning Vaiable,因为Extract Method优点很多,除了不增加临时变量增加函数长度之外,他的生命周期也比临时变量来的长,他可以让类中的所

『重构--改善既有代码的设计』读书笔记----Extract Method

在编程中,比较忌讳的一件事情就是长函数.因为长函数代表了你这段代码不能很好的复用以及内部可能出现很多别的地方的重复代码,而且这段长函数内部的处理逻辑你也不能很好的看清楚.因此,今天重构第一个手法就是处理长函数--Extract Method,抽取成一个独立的小函数. 我个人来说也很喜欢短小函数,因为他们代表了高强度的复用与灵活性.对于短小函数来说最最关键的就是短小函数的命名,其实你就是给了这些短小函数自我解释的机会,所以你如果给这些短小函数起一个接近其语义的名字,那当你读起长函数来说,就像是阅读

『重构--改善既有代码的设计』读书笔记----Change Value to Reference

有时候你会认为某个对象应该是去全局唯一的,这就是引用(Reference)的概念.它代表当你在某个地点对他进行修改之后,那么所有共享他的对象都应该在再次访问他的时候得到相应的修改.而不会像值对象(Value)一样,不可修改.举个例子,你认识小明,我也认识小明,小明忽然把头发都踢了,这个时候你认识的小明和我认识的小明都是同一个人,都是光头,这个小明就是世界的唯一实例,然而,你有100块钱,我有50块钱,我把50块钱花到只剩20,你手里的100块钱并不会因为我的50块钱改变而改变,不会相应的修改,这

『重构--改善既有代码的设计』读书笔记----Replace Method with Method Object

有时候,当你遇到一个大型函数,里面的临时变量和参数多的让你觉得根本无法进行Extract Method.重构中也大力的推荐短小函数的好处,它所带来的解释性,复用性让你收益无穷.但如果你遇到上种情况,你可能会天真的以为我只要适当的进行Replace Temp with Query,就可以把这种现象给化解.但情况往往事与愿违,不能达到你所理想的高度.这个时候你需要用到重构中的杀手锏--Replace Method with Method Object,这个手法出自Kent Beck [Beck].

『重构--改善既有代码的设计』读书笔记----Move Method

明确函数所在类的位置是很重要的.这样可以避免你的类与别的类有太多耦合.也会让你的类的内聚性变得更加牢固,让你的整个系统变得更加整洁.简单来说,如果在你的程序中,某个类的函数在使用的过程中,更多的是在和别的类进行交互,调用后者或者被后者调用,那么你就要注意了,你要去判断这个类是否真正适合他原来所在的类. 简单来说,这套手法就是在该函数最常引用的新类中建立一个有着类似行为的新函数,让旧函数变成一个单纯的委托函数或者完全删掉. Move Method是重构理论的支柱.如果一个类的责任太多,或者一个类和

『重构--改善既有代码的设计』读书笔记----代码坏味道【3】

星期六了,适当出去放松了下,回来继续我们重构的话题.今天是坏味道[3]了,很多朋友跟我私信,叫我把坏味道出完,再出手法.其实这是有道理的,很多时候,"发现"远比"怎么做"重要的多.就拿设计模式来讲,GoF里面的设计模式相信有很多人都了解过.具体的设计模式应该怎么实现啊相信有很多人都背的滚瓜烂熟,但问题的难点往往在于你应该什么时候用这个设计模式.重构也一样,手法步骤都是死的,关键在于应该发现什么时候应该重构.所以,我还是决定继续出坏味道,把坏味道全部出完我们再去学手法

『重构--改善既有代码的设计』读书笔记----Extract Class

在面向对象中,对于类这个概念我们应该有一个清晰的责任认识,就是每个类应该只有一个变化点,每个类的变化应该只受到单一的因素,即每个类应该只有一个明确的责任.当然了,说时容易做时难,很多人可能都会和我一样,一开始建立类的时候信心满满,牢记SRP原则,但随着开发进度的不断进行,很有可能你会给你原本设计好的类增加新字段或者增加新函数,对于少量的增加你可能会因为麻烦,考虑不去单独做一个新类来分解.久而久之,你这个类会变得越来越臃肿,所掌管的责任也会越来越多.这样的类往往还有大量的数据和函数,往往太大而不易

『重构--改善既有代码的设计』读书笔记----Replace Data Value with Object

当你在一个类中使用字段的时候,发现这个字段必须要和其他数据或者行为一起使用才有意义.你就应该考虑把这个数据项改成对象.在开发初期,我们对于新类中的字段往往会采取简单的基本类型形式来保存,但随着我们开发进度的增加,这些简单的数据项就不再那么简单了.比如一开始你会使用一个字符串来表示一串电话号码,但是随后你会发现,这个电话号码已经变的不再纯粹,它可能还需要“格式化”,“抽取取号”等特殊行为.一开始你可能会不以为意,觉得这个数据项就这么一两个,不会对你造成影响.但重复代码(Duplicate Code

『重构--改善既有代码的设计』读书笔记----Substitute Algorithm

重构可以把复杂的东西分解成一个个简单的小块.但有时候,你必须壮士断腕删掉整个算法,用简单的算法来取代,如果你发现做一件事情可以有更清晰的方式,那你完全有理由用更清晰的方式来解决问题.如果你开始使用程序库,发现其中库提供的功能特性和你的代码重复,那么你也应该改变你原来的算法.或者当你想要修改原先的算法,让他去做一件和原先略有差异的事情,这时候你也可以把原先的算法替换成一个较易修改的算法,让后续修改来的简单点. 使用这个手法之前,确保自己已经充分了解原先函数,替换巨大而复杂的算法是很复杂的,你可以先