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

时间:2021-08-05 16:57:34

如果你有一个数组,其中的元素各自代表不同东西,比如你有一个

QList<QString> strList;

其中strList[0]代表选手姓名,strList[1]代表选手家庭住址,很显然这个数组表示的含义已经太多,你需要用对象来替换数组,并且对于数组中的每个元素,以一个字段来表示。

数组是一种常见的用以组织数据的数据结构,不过,它们应该只用于“以某种顺序容纳一组相似对象”。对于上面的例子你可以看到一个数组容纳了不同对象,这会给使用数组的客户带来麻烦,因为他们很难记住数组的第一个元素是姓名,第二个元素是家庭住址。对象就不同了,你可以运用字段名称函数名称来表达这样的信息,你不需要死记硬背更加不需要依赖注释,而且使用对象的好处更是可以让你使用Move Method给他添加跟数据有关的行为让这个类本身越来越有魅力。

  • 做法:
  • 新建一个类用来表示数组所拥有的信息,并先以public字段来保存原先数组。
  • 修改数组的所有客户,让他们改用新类实例。
  • 编译,测试。
  • 逐一为数组元素添加取值/设值函数,根据元素的用途,为这些访问函数命名,修改客户端代码,让他们通过访问函数去取用数组元素,每次修改之后,编译测试。
  • 当所有对数组的直接访问都转而调用访问函数之后,将新类中保存该字段从public改为private.
  • 编译。
  • 对于数组中的每一个元素,在新类中创建一个类型相当的字段,修改该元素的访问函数,令他改用上述新字段。
  • 每修改一个元素,编译并测试。
  • 数组的所有元素都有了相应字段之后,删除该数组。

例子:

QList<QString> row;

row.append("Livepool");
row.append("China"); QString name = row[];
QString nation = row[];

这个数组有两个元素,其中第一个元素保存姓名,第二个元素保存国籍,很显然这样写,会困扰客户,所以我们需要重构,首先我们声明一个类用来做数组转移

class Performance
{
public:
QList<QString> m_data;
};

首先我们先把这个数组声明为public,放心,这只是暂时的,后期我们会把他设为private : ) ,现在我们要找到创建和访问数组的地方,在创建地点,我们将他改为下面代码

Performance row;

row.m_data.append("Livepool");
row.m_data.append("China"); QString name = row.m_data[];
QString nation = row.m_data[];

我们已经完成了第一步,已经把这个新建类开始引入了,别急,重构就是一步一步慢慢来,可以保证不会容易出错。接下来我们要为数组元素逐一加上有意义的设值和取值函数,首先我们从姓名开始

class Performance
{
public:
QString name() const
{
return m_data[];
} void setName(const QString &value)
{
m_data[] = value;
}
};

然后修改使用row对象的代码,让他们转而使用这些函数

row.setName("Livepool");
row.m_data[] = "China"; QString name = row.name();
QString nation = performance.m_data[];

接下来我们如法炮制第二个元素,最终代码为

class Performance
{
public:
QString nation() const
{
return m_data[];
} void setNation(const QString &value)
{
m_data[] = value;
}
}; row.setName("Livepool");
row.setNation("China"); QString name = row.name();
QString nation = row.nation();

处理完所有元素之后我们就可以放心的把字段改为private了

private:
QList<QString> m_data;

现在我们完成了重构最重要的部分----接口,但是我们还要进行替换对象内数组的过程,我可以针对每个数组元素,在类中新建一个类型相当的字段,然后修改访问数组元素的访问函数,令他直接访问新字段,从而完全摆脱对数组的依赖。

class Performance
{
public:
QString name() const
{
return m_name;
} void setName(const QString &value)
{
m_name = value;
}
private:
QString m_name;
};

待所有元素处理完毕之后,我们就可以将数组从Performance类中删除了。