Java設(shè)計(jì)模式之策略模式詳解和示例
1. 什么是策略模式
策略模式就是一種行為可能會(huì)因?yàn)椴煌倪壿嬙斐啥鄠€(gè)算法。比如人吃飯,美國(guó)人吃飯用刀叉,中國(guó)吃飯用筷子。都是吃飯的行為但是使用的工具(算法)不一樣。而策略行為就是把這寫工具封裝為對(duì)象,不同的人會(huì)動(dòng)態(tài)的調(diào)用不同的對(duì)象,來實(shí)現(xiàn)吃飯這一行為。 ?
2. 策略模式組成
環(huán)境類(Context):用一個(gè)ConcreteStrategy對(duì)象來配置。維護(hù)一個(gè)對(duì)Strategy對(duì)象的引用??啥x一個(gè)接口來讓Strategy訪問它的數(shù)據(jù)。
抽象策略類(Strategy):定義所有支持的算法的公共接口。 Context使用這個(gè)接口來調(diào)用某ConcreteStrategy定義的算法。
具體策略類(ConcreteStrategy):以Strategy接口實(shí)現(xiàn)某具體算法。
3. 策略模式UML圖

4. 示例
就按照上面提到的例子,按照策略模式UML圖按照策略模式實(shí)現(xiàn)人吃飯這件事情。
4.1 創(chuàng)建Strategy抽象策略類
public interface Strategy {
/**
* 人類怎么吃飯的
*/
public void eat();
}
4.2 創(chuàng)建環(huán)境類
public class Context {
private Strategy strategy;
public Context(){
}
public Context(Strategy strategy){
this.strategy = strategy;
}
public void setStrategy(Strategy strategy){
this.strategy = strategy;
}
public void eat(){
strategy.eat();
}
}上面環(huán)境類我根據(jù)我是實(shí)現(xiàn)了兩種方式傳入策略類,可以根據(jù)自己不同的需求就行靈活應(yīng)用給
4.3 創(chuàng)建實(shí)現(xiàn)
/**
* 美國(guó)人 吃飯
*/
@Component
public class StrategyImplOne implements Strategy{
@Override
public void eat() {
System.out.println("美國(guó)人用刀叉吃飯");
}
}/**
* 中國(guó)人 吃飯
*/
@Component
public class StrategyImplTwo implements Strategy{
@Override
public void eat() {
System.out.println("中國(guó)人用筷子吃飯");
}
}4.4 測(cè)試
public class MainTest {
public static void test(){
Context context = new Context();
context.setStrategy(new StrategyImplOne());
context.eat();
context.setStrategy(new StrategyImplTwo());
context.eat();
}
public static void main(String[] args) {
test();
}
}當(dāng)然了這是簡(jiǎn)單的場(chǎng)景,在復(fù)雜的場(chǎng)景中我們不想經(jīng)過判斷他是美國(guó)人還是中國(guó)人來創(chuàng)建對(duì)應(yīng)的實(shí)現(xiàn),設(shè)置到環(huán)境類中,我們可以給每個(gè)實(shí)現(xiàn)設(shè)置一個(gè)屬性標(biāo)識(shí),來標(biāo)識(shí)他是中國(guó)人還是美國(guó)人,然后根據(jù)標(biāo)識(shí)通過反射創(chuàng)建對(duì)應(yīng)的實(shí)現(xiàn)對(duì)象,再設(shè)置到環(huán)境類中。下面我們看結(jié)果。 ?
首先我們加入一個(gè)枚舉類(enum) 用來存儲(chǔ)我們實(shí)現(xiàn)的策略類
public enum ManType {
AMERICAN(1,"American",StrategyImplOne.class),
CHINESE(1,"Chinese",StrategyImplTwo.class)
;
private Integer code;
private String type;
private Class classModel;
// 定義一個(gè)Map 開始的時(shí)候就把這信息存儲(chǔ)到這里面
private static HashMap<String,Class> map = new HashMap<>(16);
static {
for ( ManType manType:ManType.values()) {
map.put(manType.getType(),manType.getClassModel());
}
}
ManType(Integer code, String type, Class classModel) {
this.code = code;
this.type = type;
this.classModel = classModel;
}
public Integer getCode() {
return code;
}
public String getType() {
return type;
}
public Class getClassModel() {
return classModel;
}
// 這個(gè)方法根據(jù)type獲取指定的類 通過反射來 創(chuàng)建對(duì)象
public static Class getManType(String type){
return map.get(type);
}
}創(chuàng)建好枚舉類后,我們的測(cè)試方法就變?yōu)橄旅娴倪@個(gè)樣子。寫的有點(diǎn)粗糙,大家見諒。 ?
public class MainTest {
public void test(String manType){
Context context = new Context();
try {
Strategy strategy = (Strategy) (ManType.getManType(manType).newInstance());
context.setStrategy(strategy);
context.eat();
} catch (InstantiationException e) {
e.printStackTrace();
} catch (IllegalAccessException e) {
e.printStackTrace();
}
}
public static void main(String[] args) {
new MainTest().test("American");
}
}其實(shí)也可以不用反射,如果項(xiàng)目是使用的是SpringBoot開發(fā)的話 我們可以把實(shí)現(xiàn)策略類在項(xiàng)目啟動(dòng)時(shí)就給通過@Component 給注冊(cè)到Spring容器中,然后通過ApplicationContext 從容器中獲取指定的對(duì)象。這樣也是可以的。
我就不一一展示了,感興趣的話可以自己在下面試一下。 ?
優(yōu)點(diǎn)和缺點(diǎn)
優(yōu)點(diǎn):
1、策略模式符合開閉原則。
2、避免使用多重條件轉(zhuǎn)移語(yǔ)句,如if…else…語(yǔ)句、switch 語(yǔ)句
3、使用策略模式可以提高算法的保密性和安全性。
缺點(diǎn):
1、客戶端必須知道所有的策略,并且自行決定使用哪一個(gè)策略類。
2、代碼中會(huì)產(chǎn)生非常多策略類,增加維護(hù)難度。
3、如果把策略類初始化時(shí)就放到Spring容器中,當(dāng)策略類太多事也是一種性能消耗。
到此這篇關(guān)于Java設(shè)計(jì)模式之策略模式詳解和示例的文章就介紹到這了,更多相關(guān)Java策略模式內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
MyBatis自定義映射關(guān)系和關(guān)聯(lián)查詢實(shí)現(xiàn)方法詳解
這篇文章主要介紹了MyBatis自定義映射關(guān)系和關(guān)聯(lián)查詢實(shí)現(xiàn)方法,當(dāng)POJO屬性名與數(shù)據(jù)庫(kù)列名不一致時(shí),需要自定義實(shí)體類和結(jié)果集的映射關(guān)系,在MyBatis注解開發(fā)中,使用@Results定義并使用自定義映射,使用 @ResultMap使用自定義映射2023-04-04
Java實(shí)現(xiàn)Excel百萬級(jí)數(shù)據(jù)導(dǎo)入功能的示例代碼
這篇文章主要為大家詳細(xì)介紹了Java如何實(shí)現(xiàn)Excel百萬級(jí)數(shù)據(jù)導(dǎo)入功能,文中的示例代碼講解詳細(xì),具有一定的借鑒價(jià)值,有需要的小伙伴可以參考下2024-04-04
Spring?Boot?整合?Reactor實(shí)例詳解
這篇文章主要為大家介紹了Spring?Boot?整合?Reactor實(shí)例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-09-09
Mybatis在sqlite中無法讀寫byte[]類問題的解決辦法
這篇文章主要給大家介紹了關(guān)于Mybatis在sqlite中無法讀寫byte[]類問題的解決辦法,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2020-10-10
Java如何使用SSLContext請(qǐng)求https鏈接
這篇文章主要介紹了Java如何使用SSLContext請(qǐng)求https鏈接問題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2024-01-01
JAVA調(diào)用Deepseek的api完成基本對(duì)話簡(jiǎn)單代碼示例
這篇文章主要介紹了JAVA調(diào)用Deepseek的api完成基本對(duì)話的相關(guān)資料,文中詳細(xì)講解了如何獲取DeepSeek?API密鑰、添加HTTP客戶端依賴、創(chuàng)建HTTP請(qǐng)求并使用示例代碼來對(duì)接DeepSeek?API,需要的朋友可以參考下2025-02-02

