前言
在以前写关于设计模式的博客时候,我会千篇一律的把的定义,类图以及从0到1的改造过程设计出来,但是自己每次再回顾的时候反而记得不是很牢固,现在,我更希望实际应用出发,让大家体会到设计模式的巧妙之处。
应用
在工作中,对于策略模式的应用还是比较广泛的,比如,之前在研究爬虫的时候,通过http请求获取到一些数据之后,我们需要去进行解析,得到其中想要的数据。当然,http的response响应大多都分为两类(在我工作的范围内), 基本都是html
和 Json
,对于这两种不同的类型,在我们的系统中就是利用策略模式去进行解决的,往往对于同一个流程,我们在局部的处理会有多种方式,此时,我们可以用策略模式去解决,将这个“局部”共性上提,所谓"上提", 便就要请出接口了,我们下面慢慢体会。
直接上代码
对于解析的两种方式,我们必须有两个具体解析的类
代码1
class HtmlParser {
public void parse() {
System.out.println("使用html解析方式解析数据");
}
}
class JsonParser {
public void parse() {
System.out.println("使用Json解析方式解析数据");
}
}
在解析流程中调用这两种方式,如果不用策略模式的话,一般都是如下所写的方式
代码2
class ParserProcess {
private HtmlParser htmlParser;
private JsonParser jsonParser;
public void process(String data){
if(data.equals("html数据")){
htmlParser.parse();
}else if(data.equals("json数据")){
jsonParser.parse();
}
}
}
然后我们在自己的主方法里通过调用这些方法来进行解析
代码3
public class Test {
public static void main(String[] args) {
String data = "data";
ParserProcess parser = new ParserProcess();
parser.process(data);
}
}
这样写,有个致命的缺点,先说一下前提,代码1、代码2这两块代码很可能是某个jar里提供的,我们可以选择去调用,但是无法修改(修改可能要重写)。所以,不符合开闭原则,假如现在爬到了除了html和json以外的其他数据,在拓展的时候需要修改源代码。我们现在可以通过改造来解决它的痛点
首先就是共性上提,我们定义一个接口在ParserProcess
里通过调用接口的方法,而不是直接利用实现类。
修改代码1
interface ParserMode {
void parse();
}
class HtmlParser implements ParserMode {
@Override
public void parse() {
System.out.println("使用html解析方式解析数据");
}
}
class JsonParser implements ParserMode {
@Override
public void parse() {
System.out.println("使用Json解析方式解析数据");
}
}
修改代码2
class ParserProcess {
private ParserMode concreteParser;
public void setConcreteParser(ParserMode concreteParser) {
this.concreteParser = concreteParser;
}
public void parseData(String data){
concreteParser.parse();
}
}
修改代码3
public class Test {
public static void main(String[] args) {
String data = "data";
ParserProcess parser = new ParserProcess();
// 使用html解析器解析
parser.setConcreteParser(new HtmlParser());
parser.parseData(data);
// 使用json解析器解析
parser.setConcreteParser(new JsonParser());
parser.parseData(data);
}
}
这里我们可以在运行时动态的修改它的解析方式
,这也是策略模式最大的优点之一,现在如果有了新的数据类型,我们可以直接在代码3里进行修改,而不必去修改已经固定的流程,只需实现接口。
class OtherParser implements ParserMode {
@Override
public void parse() {
System.out.println("使用其他方式解析数据");
}
}
public class Test {
public static void main(String[] args) {
String data = "data";
ParserProcess parser = new ParserProcess();
// 使用其他方式解析数据
parser.setConcreteParser(new OtherParser());
parser.parseData(data);
}
}