如果你有一个数组,其中的元素各自代表不同东西,比如你有一个
QList<QString> strList;
其中strList[0]代表选手姓名,strList[1]代表选手家庭住址,很显然这个数组表示的含义已经太多,你需要用对象来替换数组,并且对于数组中的每个元素,以一个字段来表示。
数组是一种常见的用以组织数据的数据结构,不过,它们应该只用于“以某种顺序容纳一组相似对象”。对于上面的例子你可以看到一个数组容纳了不同对象,这会给使用数组的客户带来麻烦,因为他们很难记住数组的第一个元素是姓名,第二个元素是家庭住址。对象就不同了,你可以运用字段名称和函数名称来表达这样的信息,你不需要死记硬背更加不需要依赖注释,而且使用对象的好处更是可以让你使用Move Method给他添加跟数据有关的行为让这个类本身越来越有魅力。
- 做法:
- 新建一个类用来表示数组所拥有的信息,并先以public字段来保存原先数组。
- 修改数组的所有客户,让他们改用新类实例。
- 编译,测试。
- 逐一为数组元素添加取值/设值函数,根据元素的用途,为这些访问函数命名,修改客户端代码,让他们通过访问函数去取用数组元素,每次修改之后,编译测试。
- 当所有对数组的直接访问都转而调用访问函数之后,将新类中保存该字段从public改为private.
- 编译。
- 对于数组中的每一个元素,在新类中创建一个类型相当的字段,修改该元素的访问函数,令他改用上述新字段。
- 每修改一个元素,编译并测试。
- 数组的所有元素都有了相应字段之后,删除该数组。
例子:
QList<QString> row; row.append("Livepool"); row.append("China"); QString name = row[0]; QString nation = row[1];
这个数组有两个元素,其中第一个元素保存姓名,第二个元素保存国籍,很显然这样写,会困扰客户,所以我们需要重构,首先我们声明一个类用来做数组转移
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[0]; QString nation = row.m_data[1];
我们已经完成了第一步,已经把这个新建类开始引入了,别急,重构就是一步一步慢慢来,可以保证不会容易出错。接下来我们要为数组元素逐一加上有意义的设值和取值函数,首先我们从姓名开始
class Performance { public: QString name() const { return m_data[0]; } void setName(const QString &value) { m_data[0] = value; } };
然后修改使用row对象的代码,让他们转而使用这些函数
row.setName("Livepool"); row.m_data[1] = "China"; QString name = row.name(); QString nation = performance.m_data[1];
接下来我们如法炮制第二个元素,最终代码为
class Performance { public: QString nation() const { return m_data[1]; } void setNation(const QString &value) { m_data[1] = 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类中删除了。