前言
在以前寫關(guān)于設(shè)計(jì)模式的博客時(shí)候,我會千篇一律的把的定義,類圖以及從0到1的改造過程設(shè)計(jì)出來的猛,但是自己每次再回顧的時(shí)候反而記得不是很牢固,現(xiàn)在想虎,我更希望實(shí)際應(yīng)用出發(fā),讓大家體會到設(shè)計(jì)模式的巧妙之處叛拷。
應(yīng)用
在工作中舌厨,對于策略模式的應(yīng)用還是比較廣泛的,比如忿薇,之前在研究爬蟲的時(shí)候裙椭,通過http請求獲取到一些數(shù)據(jù)之后躏哩,我們需要去進(jìn)行解析,得到其中想要的數(shù)據(jù)揉燃。當(dāng)然扫尺,http的response響應(yīng)大多都分為兩類(在我工作的范圍內(nèi)), 基本都是html
和 Json
,對于這兩種不同的類型炊汤,在我們的系統(tǒng)中就是利用策略模式去進(jìn)行解決的正驻,往往對于同一個(gè)流程,我們在局部的處理會有多種方式抢腐,此時(shí)姑曙,我們可以用策略模式去解決,將這個(gè)“局部”共性上提迈倍,所謂"上提", 便就要請出接口了伤靠,我們下面慢慢體會。
直接上代碼
對于解析的兩種方式啼染,我們必須有兩個(gè)具體解析的類
代碼1
class HtmlParser {
public void parse() {
System.out.println("使用html解析方式解析數(shù)據(jù)");
}
}
class JsonParser {
public void parse() {
System.out.println("使用Json解析方式解析數(shù)據(jù)");
}
}
在解析流程中調(diào)用這兩種方式宴合,如果不用策略模式的話,一般都是如下所寫的方式
代碼2
class ParserProcess {
private HtmlParser htmlParser;
private JsonParser jsonParser;
public void process(String data){
if(data.equals("html數(shù)據(jù)")){
htmlParser.parse();
}else if(data.equals("json數(shù)據(jù)")){
jsonParser.parse();
}
}
}
然后我們在自己的主方法里通過調(diào)用這些方法來進(jìn)行解析
代碼3
public class Test {
public static void main(String[] args) {
String data = "data";
ParserProcess parser = new ParserProcess();
parser.process(data);
}
}
這樣寫迹鹅,有個(gè)致命的缺點(diǎn)卦洽,先說一下前提,代碼1徒欣、代碼2這兩塊代碼很可能是某個(gè)jar里提供的逐样,我們可以選擇去調(diào)用,但是無法修改(修改可能要重寫)打肝。所以脂新,不符合開閉原則,假如現(xiàn)在爬到了除了html和json以外的其他數(shù)據(jù)粗梭,在拓展的時(shí)候需要修改源代碼争便。我們現(xiàn)在可以通過改造來解決它的痛點(diǎn)
首先就是共性上提,我們定義一個(gè)接口在ParserProcess
里通過調(diào)用接口的方法断医,而不是直接利用實(shí)現(xiàn)類滞乙。
修改代碼1
interface ParserMode {
void parse();
}
class HtmlParser implements ParserMode {
@Override
public void parse() {
System.out.println("使用html解析方式解析數(shù)據(jù)");
}
}
class JsonParser implements ParserMode {
@Override
public void parse() {
System.out.println("使用Json解析方式解析數(shù)據(jù)");
}
}
修改代碼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);
}
}
這里我們可以在運(yùn)行時(shí)動(dòng)態(tài)的修改它的解析方式
,這也是策略模式最大的優(yōu)點(diǎn)之一鉴嗤,現(xiàn)在如果有了新的數(shù)據(jù)類型斩启,我們可以直接在代碼3里進(jìn)行修改,而不必去修改已經(jīng)固定的流程醉锅,只需實(shí)現(xiàn)接口兔簇。
class OtherParser implements ParserMode {
@Override
public void parse() {
System.out.println("使用其他方式解析數(shù)據(jù)");
}
}
public class Test {
public static void main(String[] args) {
String data = "data";
ParserProcess parser = new ParserProcess();
// 使用其他方式解析數(shù)據(jù)
parser.setConcreteParser(new OtherParser());
parser.parseData(data);
}
}