测试工具中的设计模式实例谈 - 建造模式(Builder)
道生一,一生万。
摘要:
本文将继续以OPENCSV为案例,介绍建造模式(Builder Pattern)以及在OPENCSV中的简化模式。
1. 建造模式模式简介
建造模式属于创建类型的设计模式。其余创建型模式还有工厂模式、单例模式、原型模式等,顾名思义,就是当需要new一些对象的实例时,可以考虑使用这些类型的设计模式。 和同类型的模式相比,建造模式的特点就是“将一个复杂对象的构建与它的表示分离,使得同样的构建过程可以创建不同的表示”。其类图表示如下
9365c57354a290e1a58557aefaa1fc3f.png为了创建Product对象,定义一个Builder接口,并且在接口中定义了创建Product各个组成部分的方法BuildPart(可以有多个,负责不同部分的构建)。ConcreteBuilder则是具体的Builder类的实现。而Director类使用Builder接口的具体对象,来完成某个具体Product产品的构建过程(for循环部分)。
2. 建造模式在Opencsv中的应用
1) CsvToBeanBuilder与CsvToBean
在OpenCSV中也广泛用到了这一模式。举例来说,CsvToBean<T>是OpenCSV这个工具的核心类之一。顾名思义,其主要职责就是负责将CSV数据解析成为对应的T类型的对象(bean)。为了能够顺利完成这一工作,在初始化该类时,需要配置许多参数,如匹配策略MappingStrategy,对应的Bean类等等。OpenCSV提供了CsvToBeanBuilder这一建造类,方便完成这个初始化的工作。这个类的部分方法如下图所示,
4a961bef9be2af78aa36b20cbcb5e38e.png其中的各种以with开头的方法就是完成CsvToBean对象的某个组成部分的构建任务,如withSeparator(char)就是指定CSV数据的分隔符号,如常见的","或者“;”。
2) 测试用例(Director)
在定义了构建类和具体的构建方法后,就可以在外部通过Director完成产品CsvToBean的构建过程了。在OpenCSV提供的单元测试用例中,给出了如下的使用案例,
@Test
**public** **void** testColumnMappingStrategyWithBuilder() **throws** FileNotFoundException {
List<AnnotatedMockBeanFull> result =
**new** CsvToBeanBuilder<AnnotatedMockBeanFull>(**new** FileReader("src/test/resources/testinputposfullgood.csv"))
.withSeparator(';')
.withType(AnnotatedMockBeanFull.**class**)
.build()
.parse();
*assertEquals*(2, result.size());
}
通过调用CsvToBeanBuilder提供的各个withXXX构建方法以及build方法,构造出了一个匿名的CsvToBean实例,并通过调用其parse()方法完成对提供的.csv文件的解析,将其每一行数据解析成为一个AnnotatedMockBeanFull类的实例,并存放进 result中。
3) 案例点评
在这个案例中,建造模式简化成为了如下的形式,
f507d980b882ed0c11d928f901402d97.jpg即,去掉了Builder接口,直接定义了CsvToBeanBuilder这一ConcreateBuilder类,来对需要构建的产品Product产品,即CsvToBean进行构建。这一过程的的导演Director,则由具体的测试用例类在工具外部完成。这样,对于灵活多样的CSV数据文件,可以建造出来各种不同的Product来实施解析,提高了工具使用的便利性。在CsvToBean中,之前存在着许多重载的parse方法,实现对CSV文件的不同方式的解析。而通过使用CsvToBeanBuilder之后,则只要通过其构造出所需要的CsvToBean实例,并调用无参的parse方法即可完成。
b727f9f972b8d7f6463a0e7d7b1715ee.png
从而屏蔽了具体的parse实现,实现了解析过程和数据内容和方式的解耦合。
类似的,Opencsv中还使用了CSVParserBuilder、CSVReaderBuilder等来建造类来提供CSVParser、CSVReader等类的实例,也都是采用了前述介绍的简化模式。感兴趣的读者可以阅读OPENCSV的源码进一步了解。