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() + );
} class MfDateSub...
Date *nextDate()
{
return new Date(getYear(), getMonth(), getDate() + );
}

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

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() + );
}

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

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);

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

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

  1. 『重构--改善既有代码的设计』读书笔记----Introduce Foreign Method

    当你无法获得一个类的源代码或者没有权限去修改这个类的时候,你对于这种为你服务的类,你可能会出现需要别的需求的时候,比如一个Date类,你需要能够让他本身直接返回出他的后一天的对象,但他没有,这个时候你 ...

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

    有时候你会遇到一系列复杂的表达式连续运算的时候,这个时候你可能根本招架不住如此长或者是如此复杂的长函数.这个时候你可以通过引用临时变量来储存他们的结果,将这些长函数的结果分成一个个临时变量来让函数清晰 ...

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

    在编程中,比较忌讳的一件事情就是长函数.因为长函数代表了你这段代码不能很好的复用以及内部可能出现很多别的地方的重复代码,而且这段长函数内部的处理逻辑你也不能很好的看清楚.因此,今天重构第一个手法就是处 ...

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

    有时候你会认为某个对象应该是去全局唯一的,这就是引用(Reference)的概念.它代表当你在某个地点对他进行修改之后,那么所有共享他的对象都应该在再次访问他的时候得到相应的修改.而不会像值对象(Va ...

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

    有时候,当你遇到一个大型函数,里面的临时变量和参数多的让你觉得根本无法进行Extract Method.重构中也大力的推荐短小函数的好处,它所带来的解释性,复用性让你收益无穷.但如果你遇到上种情况,你 ...

  6. 『重构--改善既有代码的设计』读书笔记---Duplicate Observed Data

    当MVC出现的时候,极大的推动了Model与View分离的潮流.然而对于一些已存在的老系统或者没有维护好的系统,你都会看到当前存在大把的巨大类----将Model,View,Controller都写在 ...

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

    如果你有一个数组,其中的元素各自代表不同东西,比如你有一个 QList<QString> strList; 其中strList[0]代表选手姓名,strList[1]代表选手家庭住址,很显 ...

  8. 『重构--改善既有代码的设计』读书笔记----Self Encapsulate Field

    如果你直接访问一个字段,你就会和这个字段直接的耦合关系变得笨拙.也就是说当这个字段权限更改,或者名称更改之后你的客户端代码都需要做相应的改变,此时你可以为这个字段建立设值和取值函数并且只以这些函数来访 ...

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

    明确函数所在类的位置是很重要的.这样可以避免你的类与别的类有太多耦合.也会让你的类的内聚性变得更加牢固,让你的整个系统变得更加整洁.简单来说,如果在你的程序中,某个类的函数在使用的过程中,更多的是在和 ...

随机推荐

  1. JQuery中如何click中传递参数

    代码如下: click(data,fn)中的data其实是json对象,取的时候,只能通过当前的事件源来取,data是默认放在event中的,所以这里的data是eventdata,引用的时候也使用e ...

  2. Rest中的XML与JSON的序列化与反序列化

    #region 获取XML的序列化和反序列化 /// <summary> /// 对象进行序列化生成XML /// </summary> /// <typeparam n ...

  3. [JIT_APP]Java基础知识总结

    一.Java语言的基础知识 1. 开发Java语言的公司 美国Sun(Sum Microsystems)公司开发.   2.Java的3个版本 J2SE(Java2 Standard Edition) ...

  4. MenuDrawer的使用

    ---恢复内容开始--- MenuDrawer框架是一个可以实现上下左右滑动的框架,在使用中可以在xml文件中配置也可以在java代码中实现效果的配置 可以以jar的形式或依赖的形式存在      用 ...

  5. 配置java软件

    到官网下载了jdk和jre的安装包 配置时发现javac显示 "javac不是内部或外部命令,等等一些的问题" 上百度搜索后,发现是win10系统配置path的时候需要写绝对路径而 ...

  6. (C#)与Windows用户账户信息的获取

    Console.WriteLine(Environment.UserName); //计算机NetBIOS名称 Console.WriteLine(Environment.MachineName); ...

  7. 1040. Longest Symmetric String (25)

    题目链接:http://www.patest.cn/contests/pat-a-practise/1040 题目: 1040. Longest Symmetric String (25) 时间限制 ...

  8. POJ1330Nearest Common Ancestors——近期公共祖先(离线Tarjan)

    http://poj.org/problem? id=1330 给一个有根树,一个查询节点(u,v)的近期公共祖先 836K 16MS #include<iostream> #includ ...

  9. [RxJS] Aggregating Streams With Reduce And Scan using RxJS

    What is the RxJS equivalent of Array reduce? What if I want to emit my reduced or aggregated value a ...

  10. Windows内核之线程的调度,优先级,亲缘性

    1 调度 Windows不是实时操作系统,它是抢占式多线程操作系统.在如果全部优先级同样的情况下,CPU对线程的调度原则是每隔20m就会切换到下一个线程,依据Context中的IP和SP来接着运行上次 ...