在之前的文章我們已經介紹了設計模式中的建立者模式和結構型模式,下面我們來介紹最後一部分行為型模式
行為型模式用於描述程式在執行時複雜的流程控制,即描述多個類或物件之間怎樣相互共同作業共同完成單個物件都無法單獨完成的任務
行為型模式分為類行為模式和物件行為模式,前者採用繼承機制來在類間分派行為,後者採用組合或聚合在物件間分配行為。
由於組合關係或聚合關係比繼承關係耦合度低,滿足「合成複用原則」,所以物件行為模式比類行為模式具有更大的靈活性。
下面我們將介紹十一種行為型模式:
首先我們來介紹模板方法模式
首先我們給出模板方法模式的概念:
模板方法(Template Method)模式包含以下主要角色:
抽象類(Abstract Class):負責給出一個演演算法的輪廓和骨架。它由一個模板方法和若干個基本方法構成。
模板方法:定義了演演算法的骨架,按某種順序呼叫其包含的基本方法。
基本方法:是實現演演算法各個步驟的方法,是模板方法的組成部分。基本方法又可以分為三種:
抽象方法(Abstract Method) :一個抽象方法由抽象類宣告、由其具體子類實現。
具體方法(Concrete Method) :一個具體方法由一個抽象類或具體類宣告並實現,其子類可以進行覆蓋也可以直接繼承。
勾點方法(Hook Method) :在抽象類中已經實現,包括用於判斷的邏輯方法和需要子類重寫的空方法兩種。
一般勾點方法是用於判斷的邏輯方法,這類方法名一般為isXxx,返回值型別為boolean型別。
具體子類(Concrete Class):實現抽象類中所定義的抽象方法和勾點方法,它們是一個頂級邏輯的組成步驟。
我們給出一個簡單的例子來介紹模板方法模式:
具體分析:
/*
【例】炒菜
炒菜的步驟是固定的,分為倒油、熱油、倒蔬菜、倒調料品、翻炒等步驟。現通過模板方法模式來用程式碼模擬。
上述的AbstractClass就是抽象類,我們在抽象類給出一個模板方法cookProcess,裡面會給出其他基本方法的執行順序,部分基本方法會有具體內容,部分基本方法屬於Abstract方法,由子類去實現
下面的ConcreteClass_BaoCai和ConcreteClass_CaiXin屬於子類實現類,他們會繼承父類別的模板方法,同時重寫抽象基本方法完成自己的需求
*/
/* 具體程式碼 */
// 抽象類
public abstract class AbstractClass {
// 模板方法(為防止惡意操作,一般模板方法都加上 final 關鍵詞)
public final void cookProcess() {
//第一步:倒油
this.pourOil();
//第二步:熱油
this.heatOil();
//第三步:倒蔬菜
this.pourVegetable();
//第四步:倒調味料
this.pourSauce();
//第五步:翻炒
this.fry();
}
// 下述均為基本方法
public void pourOil() {
System.out.println("倒油");
}
//第二步:熱油是一樣的,所以直接實現
public void heatOil() {
System.out.println("熱油");
}
//第三步:倒蔬菜是不一樣的(一個下包菜,一個是下菜心)
public abstract void pourVegetable();
//第四步:倒調味料是不一樣
public abstract void pourSauce();
//第五步:翻炒是一樣的,所以直接實現
public void fry(){
System.out.println("炒啊炒啊炒到熟啊");
}
}
// Baocai實現類
public class ConcreteClass_BaoCai extends AbstractClass {
@Override
public void pourVegetable() {
System.out.println("下鍋的蔬菜是包菜");
}
@Override
public void pourSauce() {
System.out.println("下鍋的醬料是辣椒");
}
}
// Caixin實現類
public class ConcreteClass_CaiXin extends AbstractClass {
@Override
public void pourVegetable() {
System.out.println("下鍋的蔬菜是菜心");
}
@Override
public void pourSauce() {
System.out.println("下鍋的醬料是蒜蓉");
}
}
public class Client {
public static void main(String[] args) {
//炒手撕包菜
ConcreteClass_BaoCai baoCai = new ConcreteClass_BaoCai();
baoCai.cookProcess();
//炒蒜蓉菜心
ConcreteClass_CaiXin caiXin = new ConcreteClass_CaiXin();
caiXin.cookProcess();
}
}
首先我們給出模板方法模式的適用場景:
然後我們給出模板方法模式的優點:
提高程式碼複用性
將相同部分的程式碼放在抽象的父類別中,而將不同的程式碼放入不同的子類中。
實現了反向控制
通過一個父類別呼叫其子類的操作,通過對子類的具體實現擴充套件不同的行為,實現了反向控制 ,並符合「開閉原則」。
最後我們給出模板方法模式的缺點:
下面我們來介紹策略模式
首先我們給出策略模式的概念:
策略模式和模板模式其實比較相似,只不過前者使用聚合,後者使用繼承。
該模式定義了一系列演演算法,並將每個演演算法封裝起來,使它們可以相互替換,且演演算法的變化不會影響使用演演算法的客戶。
物件行為模式,它通過對演演算法進行封裝,把使用演演算法的責任和演演算法的實現分割開來,並委派給不同的物件對這些演演算法進行管理。
我們給出一個簡單的例子說明:
策略模式的主要角色如下:
我們同樣給出一個簡單的案例講解策略模式:
具體分析:
/*
【例】促銷活動
一家百貨公司在定年度的促銷活動。針對不同的節日(春節、中秋節、聖誕節)推出不同的促銷活動,由促銷員將促銷活動展示給客戶。
其中SalesMan就是環境類,Strategy是抽象策略類,下面的方案就是具體策略類
SalesMan中聚合一個Strategy,然後會用子類去填充,具有一定格式但不同實現的子類就可以不斷更替Strategy而實現策略更換
*/
/* 程式碼展示 */
// 環境類(呼叫更換策略類)
public class SalesMan {
//持有抽象策略角色的參照
private Strategy strategy;
public SalesMan(Strategy strategy) {
this.strategy = strategy;
}
//向客戶展示促銷活動
public void salesManShow(){
strategy.show();
}
}
// 抽象策略類
public interface Strategy {
void show();
}
// 具體策略類
//為春節準備的促銷活動A
public class StrategyA implements Strategy {
public void show() {
System.out.println("買一送一");
}
}
//為中秋準備的促銷活動B
public class StrategyB implements Strategy {
public void show() {
System.out.println("滿200元減50元");
}
}
//為聖誕準備的促銷活動C
public class StrategyC implements Strategy {
public void show() {
System.out.println("滿1000元加一元換購任意200元以下商品");
}
}
首先我們給出策略模式的適用場景:
然後我們給出策略模式的優點:
策略類之間可以自由切換
由於策略類都實現同一個介面,所以使它們之間可以自由切換。
易於擴充套件
增加一個新的策略只需要新增一個具體的策略類即可,基本不需要改變原有的程式碼,符合「開閉原則「
避免使用多重條件選擇語句(if else),充分體現物件導向設計思想。
最後我們給出策略模式的缺點:
下面我們來介紹命令模式
首先我們給出命令模式的概念:
我們給出一個簡單範例:
命令模式包含以下主要角色:
我們給出一個簡單的案例來介紹命令模式:
具體分析:
/*
將上面的案例用程式碼實現,那我們就需要分析命令模式的角色在該案例中由誰來充當。
服務員: 就是呼叫者角色,由她來發起命令。
資深大廚: 就是接收者角色,真正命令執行的物件。
訂單: 命令中包含訂單。
注意:
訂單就是命令,命令是自己具有的,有指定的接收物件
服務員只負責將啟動命令,實則還是呼叫命令內部的方法
*/
/* 程式碼展示 */
// 抽象命令類
public interface Command {
void execute();//只需要定義一個統一的執行方法
}
// 具體命令類
public class OrderCommand implements Command {
// 持有接受者物件(就是實現者,當呼叫者呼叫命令後實現者會去執行該命令)
private SeniorChef receiver;
// 執行的內容儲存
private Order order;
public OrderCommand(SeniorChef receiver, Order order){
this.receiver = receiver;
this.order = order;
}
// 具體的命令執行,呼叫者只負責呼叫該方法
public void execute() {
System.out.println(order.getDiningTable() + "桌的訂單:");
Set<String> keys = order.getFoodDic().keySet();
for (String key : keys) {
receiver.makeFood(order.getFoodDic().get(key),key);
}
try {
Thread.sleep(100);//停頓一下 模擬做飯的過程
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println(order.getDiningTable() + "桌的飯弄好了");
}
}
// domain實體類
public class Order {
// 餐桌號碼
private int diningTable;
// 用來儲存餐名並記錄份數
private Map<String, Integer> foodDic = new HashMap<String, Integer>();
public int getDiningTable() {
return diningTable;
}
public void setDiningTable(int diningTable) {
this.diningTable = diningTable;
}
public Map<String, Integer> getFoodDic() {
return foodDic;
}
public void setFoodDic(String name, int num) {
foodDic.put(name,num);
}
}
// 資深大廚類 是命令的Receiver
public class SeniorChef {
public void makeFood(int num,String foodName) {
System.out.println(num + "份" + foodName);
}
}
// 呼叫者,負責協調請求和接收者
public class Waitor {
private ArrayList<Command> commands;//可以持有很多的命令物件
public Waitor() {
commands = new ArrayList();
}
public void setCommand(Command cmd){
commands.add(cmd);
}
// 發出命令 喊 訂單來了,廚師開始執行
public void orderUp() {
System.out.println("美女服務員:叮咚,大廚,新訂單來了.......");
for (int i = 0; i < commands.size(); i++) {
Command cmd = commands.get(i);
if (cmd != null) {
cmd.execute();
}
}
}
}
// 測試類
public class Client {
public static void main(String[] args) {
//建立2個order
Order order1 = new Order();
order1.setDiningTable(1);
order1.getFoodDic().put("西紅柿雞蛋麵",1);
order1.getFoodDic().put("小杯可樂",2);
Order order2 = new Order();
order2.setDiningTable(3);
order2.getFoodDic().put("尖椒肉絲蓋飯",1);
order2.getFoodDic().put("小杯雪碧",1);
//建立接收者
SeniorChef receiver=new SeniorChef();
//將訂單和接收者封裝成命令物件
OrderCommand cmd1 = new OrderCommand(receiver, order1);
OrderCommand cmd2 = new OrderCommand(receiver, order2);
//建立呼叫者 waitor
Waitor invoker = new Waitor();
invoker.setCommand(cmd1);
invoker.setCommand(cmd2);
//將訂單帶到櫃檯 並向廚師喊 訂單來了
invoker.orderUp();
}
}
我們首先給出命令模式的適用場景:
然後我們給出命令模式的優點:
最後我們給出命令模式的缺點:
下面我們來介紹責任鏈模式
首先我們先來簡單介紹一下責任鏈模式:
我們給出一個簡單例子:
職責鏈模式主要包含以下角色:
我們同樣給出一個簡單案例來講解責任鏈模式:
具體分析:
/*
【例】
現需要開發一個請假流程控制系統。
請假一天以下的假只需要小組長同意即可;請假1天到3天的假還需要部門經理同意;請求3天到7天還需要總經理同意才行。
LeaveRequest:請假條,記錄任命,日期,資訊;屬於實體類
Handler:抽象處理者
Leader:具體處理者
*/
/* 程式碼展示 */
// 請假條(實體類,僅用於記錄資訊)
public class LeaveRequest {
private String name;//姓名
private int num;//請假天數
private String content;//請假內容
public LeaveRequest(String name, int num, String content) {
this.name = name;
this.num = num;
this.content = content;
}
public String getName() {
return name;
}
public int getNum() {
return num;
}
public String getContent() {
return content;
}
}
// 處理者抽象類
public abstract class Handler {
// 請假日期分界線,用於子類使用
protected final static int NUM_ONE = 1;
protected final static int NUM_THREE = 3;
protected final static int NUM_SEVEN = 7;
// 該領導處理的請假天數區間
private int numStart;
private int numEnd;
// 領導上面還有領導(責任鏈的下一位)
private Handler nextHandler;
// 設定請假天數範圍 上不封頂
public Handler(int numStart) {
this.numStart = numStart;
}
// 設定請假天數範圍
public Handler(int numStart, int numEnd) {
this.numStart = numStart;
this.numEnd = numEnd;
}
// 設定上級領導
public void setNextHandler(Handler nextHandler){
this.nextHandler = nextHandler;
}
// 提交請假條(這裡是一個提交方法,引數為leaveRequest,主要是給子類的領導層使用的)
public final void submit(LeaveRequest leave){
// 首先判斷是否請假
if(0 == this.numStart){
return;
}
//如果請假天數達到該領導者的處理要求
if(leave.getNum() >= this.numStart){
this.handleLeave(leave);
//如果還有上級 並且請假天數超過了當前領導的處理範圍
if(null != this.nextHandler && leave.getNum() > numEnd){
this.nextHandler.submit(leave);//繼續提交
} else {
System.out.println("流程結束");
}
}
}
// 各級領導處理請假條方法
protected abstract void handleLeave(LeaveRequest leave);
}
// 小組長
public class GroupLeader extends Handler {
public GroupLeader() {
//小組長處理1-3天的請假
super(Handler.NUM_ONE, Handler.NUM_THREE);
}
@Override
protected void handleLeave(LeaveRequest leave) {
System.out.println(leave.getName() + "請假" + leave.getNum() + "天," + leave.getContent() + "。");
System.out.println("小組長審批:同意。");
}
}
// 部門經理
public class Manager extends Handler {
public Manager() {
//部門經理處理3-7天的請假
super(Handler.NUM_THREE, Handler.NUM_SEVEN);
}
@Override
protected void handleLeave(LeaveRequest leave) {
System.out.println(leave.getName() + "請假" + leave.getNum() + "天," + leave.getContent() + "。");
System.out.println("部門經理審批:同意。");
}
}
// 總經理
public class GeneralManager extends Handler {
public GeneralManager() {
//部門經理處理7天以上的請假
super(Handler.NUM_SEVEN);
}
@Override
protected void handleLeave(LeaveRequest leave) {
System.out.println(leave.getName() + "請假" + leave.getNum() + "天," + leave.getContent() + "。");
System.out.println("總經理審批:同意。");
}
}
// 測試類
public class Client {
public static void main(String[] args) {
//請假條來一張
LeaveRequest leave = new LeaveRequest("小花",5,"身體不適");
//各位領導
GroupLeader groupLeader = new GroupLeader();
Manager manager = new Manager();
GeneralManager generalManager = new GeneralManager();
groupLeader.setNextHandler(manager);//小組長的領導是部門經理
manager.setNextHandler(generalManager);//部門經理的領導是總經理
//之所以在這裡設定上級領導,是因為可以根據實際需求來更改設定,如果實戰中上級領導人都是固定的,則可以移到領導實現類中。
//提交申請
groupLeader.submit(leave);
}
}
首先我們給出責任鏈模式的優點:
降低了物件之間的耦合度
該模式降低了請求傳送者和接收者的耦合度。
增強了系統的可延伸性
可以根據需要增加新的請求處理類,滿足開閉原則。
增強了給物件指派職責的靈活性
當工作流程發生變化,可以動態地改變鏈內的成員或者修改它們的次序,也可動態地新增或者刪除責任。
責任鏈簡化了物件之間的連線
一個物件只需保持一個指向其後繼者的參照,不需保持其他所有處理者的參照,這避免了使用眾多的 if 或者 if···else 語句。
責任分擔
每個類只需要處理自己該處理的工作,不能處理的傳遞給下一個物件完成,明確各類的責任範圍,符合類的單一職責原則。
然後我們給出責任鏈模式的缺點:
下面我們來介紹狀態模式
首先我們給出狀態模式的概念:
狀態模式包含以下主要角色。
我們首先給出一個非狀態模式:
具體分析:
/*
【例】通過按鈕來控制一個電梯的狀態,一個電梯有開門狀態,關門狀態,停止狀態,執行狀態。每一種狀態改變,都有可能要根據其他狀態來更新處理。例如,如果電梯門現在處於執行時狀態,就不能進行開門操作,而如果電梯門是停止狀態,就可以執行開門操作。
下述程式碼問題:
- 使用了大量的switch…case這樣的判斷(if…else也是一樣),使程式的可閱讀性變差。
- 擴充套件性很差。如果新加了斷電的狀態,我們需要修改上面判斷邏輯
*/
/* 程式碼展示 */
public interface ILift {
//電梯的4個狀態
//開門狀態
public final static int OPENING_STATE = 1;
//關門狀態
public final static int CLOSING_STATE = 2;
//執行狀態
public final static int RUNNING_STATE = 3;
//停止狀態
public final static int STOPPING_STATE = 4;
//設定電梯的狀態
public void setState(int state);
//電梯的動作
public void open();
public void close();
public void run();
public void stop();
}
public class Lift implements ILift {
private int state;
@Override
public void setState(int state) {
this.state = state;
}
//執行關門動作
@Override
public void close() {
switch (this.state) {
case OPENING_STATE:
System.out.println("電梯關門了。。。");//只有開門狀態可以關閉電梯門,可以對應電梯狀態表來看
this.setState(CLOSING_STATE);//關門之後電梯就是關閉狀態了
break;
case CLOSING_STATE:
//do nothing //已經是關門狀態,不能關門
break;
case RUNNING_STATE:
//do nothing //執行時電梯門是關著的,不能關門
break;
case STOPPING_STATE:
//do nothing //停止時電梯也是關著的,不能關門
break;
}
}
//執行開門動作
@Override
public void open() {
switch (this.state) {
case OPENING_STATE://門已經開了,不能再開門了
//do nothing
break;
case CLOSING_STATE://關門狀態,門開啟:
System.out.println("電梯門開啟了。。。");
this.setState(OPENING_STATE);
break;
case RUNNING_STATE:
//do nothing 執行時電梯不能開門
break;
case STOPPING_STATE:
System.out.println("電梯門開了。。。");//電梯停了,可以開門了
this.setState(OPENING_STATE);
break;
}
}
//執行執行動作
@Override
public void run() {
switch (this.state) {
case OPENING_STATE://電梯不能開著門就走
//do nothing
break;
case CLOSING_STATE://門關了,可以執行了
System.out.println("電梯開始執行了。。。");
this.setState(RUNNING_STATE);//現在是執行狀態
break;
case RUNNING_STATE:
//do nothing 已經是執行狀態了
break;
case STOPPING_STATE:
System.out.println("電梯開始執行了。。。");
this.setState(RUNNING_STATE);
break;
}
}
//執行停止動作
@Override
public void stop() {
switch (this.state) {
case OPENING_STATE: //開門的電梯已經是是停止的了(正常情況下)
//do nothing
break;
case CLOSING_STATE://關門時才可以停止
System.out.println("電梯停止了。。。");
this.setState(STOPPING_STATE);
break;
case RUNNING_STATE://執行時當然可以停止了
System.out.println("電梯停止了。。。");
this.setState(STOPPING_STATE);
break;
case STOPPING_STATE:
//do nothing
break;
}
}
}
public class Client {
public static void main(String[] args) {
Lift lift = new Lift();
lift.setState(ILift.STOPPING_STATE);//電梯是停止的
lift.open();//開門
lift.close();//關門
lift.run();//執行
lift.stop();//停止
}
}
然後我們給出狀態模式下的修改案例:
具體分析:
/*
對上述電梯的案例使用狀態模式進行改進
*/
/* 程式碼展示 */
//抽象狀態類
public abstract class LiftState {
//定義一個環境角色,也就是封裝狀態的變化引起的功能變化
protected Context context;
public void setContext(Context context) {
this.context = context;
}
//電梯開門動作
public abstract void open();
//電梯關門動作
public abstract void close();
//電梯執行動作
public abstract void run();
//電梯停止動作
public abstract void stop();
}
//開啟狀態
public class OpenningState extends LiftState {
//開啟當然可以關閉了,我就想測試一下電梯門開關功能
@Override
public void open() {
System.out.println("電梯門開啟...");
}
@Override
public void close() {
//狀態修改
super.context.setLiftState(Context.closeingState);
//動作委託為CloseState來執行,也就是委託給了ClosingState子類執行這個動作
super.context.getLiftState().close();
}
//電梯門不能開著就跑,這裡什麼也不做
@Override
public void run() {
//do nothing
}
//開門狀態已經是停止的了
@Override
public void stop() {
//do nothing
}
}
//執行狀態
public class RunningState extends LiftState {
//執行的時候開電梯門?你瘋了!電梯不會給你開的
@Override
public void open() {
//do nothing
}
//電梯門關閉?這是肯定了
@Override
public void close() {//雖然可以關門,但這個動作不歸我執行
//do nothing
}
//這是在執行狀態下要實現的方法
@Override
public void run() {
System.out.println("電梯正在執行...");
}
//這個事絕對是合理的,光執行不停止還有誰敢做這個電梯?!估計只有上帝了
@Override
public void stop() {
super.context.setLiftState(Context.stoppingState);
super.context.stop();
}
}
//停止狀態
public class StoppingState extends LiftState {
//停止狀態,開門,那是要的!
@Override
public void open() {
//狀態修改
super.context.setLiftState(Context.openningState);
//動作委託為CloseState來執行,也就是委託給了ClosingState子類執行這個動作
super.context.getLiftState().open();
}
@Override
public void close() {//雖然可以關門,但這個動作不歸我執行
//狀態修改
super.context.setLiftState(Context.closeingState);
//動作委託為CloseState來執行,也就是委託給了ClosingState子類執行這個動作
super.context.getLiftState().close();
}
//停止狀態再跑起來,正常的很
@Override
public void run() {
//狀態修改
super.context.setLiftState(Context.runningState);
//動作委託為CloseState來執行,也就是委託給了ClosingState子類執行這個動作
super.context.getLiftState().run();
}
//停止狀態是怎麼發生的呢?當然是停止方法執行了
@Override
public void stop() {
System.out.println("電梯停止了...");
}
}
//關閉狀態
public class ClosingState extends LiftState {
@Override
//電梯門關閉,這是關閉狀態要實現的動作
public void close() {
System.out.println("電梯門關閉...");
}
//電梯門關了再開啟,逗你玩呢,那這個允許呀
@Override
public void open() {
super.context.setLiftState(Context.openningState);
super.context.open();
}
//電梯門關了就跑,這是再正常不過了
@Override
public void run() {
super.context.setLiftState(Context.runningState);
super.context.run();
}
//電梯門關著,我就不按樓層
@Override
public void stop() {
super.context.setLiftState(Context.stoppingState);
super.context.stop();
}
}
//環境角色
public class Context {
//定義出所有的電梯狀態
public final static OpenningState openningState = new OpenningState();//開門狀態,這時候電梯只能關閉
public final static ClosingState closeingState = new ClosingState();//關閉狀態,這時候電梯可以執行、停止和開門
public final static RunningState runningState = new RunningState();//執行狀態,這時候電梯只能停止
public final static StoppingState stoppingState = new StoppingState();//停止狀態,這時候電梯可以開門、執行
//定義一個當前電梯狀態
private LiftState liftState;
public LiftState getLiftState() {
return this.liftState;
}
public void setLiftState(LiftState liftState) {
//當前環境改變
this.liftState = liftState;
//把當前的環境通知到各個實現類中
this.liftState.setContext(this);
}
public void open() {
this.liftState.open();
}
public void close() {
this.liftState.close();
}
public void run() {
this.liftState.run();
}
public void stop() {
this.liftState.stop();
}
}
//測試類
public class Client {
public static void main(String[] args) {
Context context = new Context();
context.setLiftState(new ClosingState());
context.open();
context.close();
context.run();
context.stop();
}
}
我們首先給出狀態模式的適用場景:
然後我們給出狀態模式的優點:
最後我們給出狀態模式的缺點:
下面我們來介紹觀察者模式
首先我們給出觀察者模式的概念:
在觀察者模式中有如下角色:
我們通過一個案例來解釋觀察者模式:
具體分析:
/*
【例】微信公眾號
在使用微信公眾號時,大家都會有這樣的體驗,當你關注的公眾號中有新內容更新的話,它就會推播給關注公眾號的微信使用者端。
我們使用觀察者模式來模擬這樣的場景,微信使用者就是觀察者,微信公眾號是被觀察者,有多個的微信使用者關注了這個公眾號。
其中微信公眾號就是被觀察者,被觀察者可以儲存多個觀察者,當被觀察者做出一些修改後,就會呼叫一個方法去通知觀察者並修改內容
*/
/* 程式碼展示 */
// 抽象觀察者(觀察者中具有一個修改方法,當被觀察者被修改後,會導致觀察者呼叫update方法)
public interface Observer {
void update(String message);
}
// 具體觀察者(這裡是指使用者,這裡指wx公眾號發表文章後,觀察者會收到一條提示資訊)
public class WeixinUser implements Observer {
// 微信使用者名稱
private String name;
public WeixinUser(String name) {
this.name = name;
}
@Override
public void update(String message) {
System.out.println(name + "-" + message);
}
}
// 抽象主題類(被觀察者)
public interface Subject {
// 增加訂閱者
public void attach(Observer observer);
// 刪除訂閱者
public void detach(Observer observer);
// 通知訂閱者更新訊息
public void notify(String message);
}
// 具體主題類(被觀察者)
public class SubscriptionSubject implements Subject {
// 儲存訂閱公眾號的微信使用者
private List<Observer> weixinUserlist = new ArrayList<Observer>();
@Override
public void attach(Observer observer) {
weixinUserlist.add(observer);
}
@Override
public void detach(Observer observer) {
weixinUserlist.remove(observer);
}
// 當wx公眾號發表文章,就會呼叫該方法,然後通知各個觀察者去update
@Override
public void notify(String message) {
for (Observer observer : weixinUserlist) {
observer.update(message);
}
}
}
// 使用者端資訊
public class Client {
public static void main(String[] args) {
SubscriptionSubject mSubscriptionSubject=new SubscriptionSubject();
//建立微信使用者
WeixinUser user1=new WeixinUser("孫悟空");
WeixinUser user2=new WeixinUser("豬悟能");
WeixinUser user3=new WeixinUser("沙悟淨");
//訂閱公眾號
mSubscriptionSubject.attach(user1);
mSubscriptionSubject.attach(user2);
mSubscriptionSubject.attach(user3);
//公眾號更新發出訊息給訂閱的微信使用者
mSubscriptionSubject.notify("傳智黑馬的專欄更新了");
}
}
首先我們給出觀察者模式的適用場景:
然後我們給出觀察者模式的優點:
最後我們給出觀察者模式的缺點:
下面我們來介紹中介者模式
首先我們給出中介者模式的概念:
我們給出一個簡單的範例:
中介者模式包含以下主要角色:
抽象中介者(Mediator)角色:它是中介者的介面,提供了同事物件註冊與轉發同事物件資訊的抽象方法。
具體中介者(ConcreteMediator)角色:實現中介者介面,定義一個 List 來管理同事物件,協調各個同事角色之間的互動關係,因此它依賴於同事角色。
抽象同事類(Colleague)角色:定義同事類的介面,儲存中介者物件,提供同事物件互動的抽象方法,實現所有相互影響的同事類的公共功能。
具體同事類(Concrete Colleague)角色:是抽象同事類的實現者,當需要與其他同事物件互動時,由中介者物件負責後續的互動。
我們通過一個簡單的案例來介紹中介者模式:
具體分析:
/*
【例】租房
現在租房基本都是通過房屋中介,房主將房屋託管給房屋中介,而租房者從房屋中介獲取房屋資訊。房屋中介充當租房者與房屋所有者之間的中介者。
租房者本身需要和所有房東聯絡才能獲得房屋資訊,但是可以通過中介獲得所有房屋資訊,這就實現瞭解耦操作
*/
/* 程式碼展示 */
// 抽象中介者
public abstract class Mediator {
//申明一個聯絡方法(前者是資訊,後者是資訊傳送人)
public abstract void constact(String message,Person person);
}
// 抽象同事類
public abstract class Person {
protected String name;
protected Mediator mediator;
public Person(String name,Mediator mediator){
this.name = name;
this.mediator = mediator;
}
}
// 具體同事類 房屋擁有者
public class HouseOwner extends Person {
public HouseOwner(String name, Mediator mediator) {
super(name, mediator);
}
//與中介者聯絡
public void constact(String message){
mediator.constact(message, this);
}
//獲取資訊
public void getMessage(String message){
System.out.println("房主" + name +"獲取到的資訊:" + message);
}
}
// 具體同事類 承租人
public class Tenant extends Person {
public Tenant(String name, Mediator mediator) {
super(name, mediator);
}
//與中介者聯絡
public void constact(String message){
mediator.constact(message, this);
}
//獲取資訊
public void getMessage(String message){
System.out.println("租房者" + name +"獲取到的資訊:" + message);
}
}
//中介機構
public class MediatorStructure extends Mediator {
//首先中介結構必須知道所有房主和租房者的資訊
private HouseOwner houseOwner;
private Tenant tenant;
public HouseOwner getHouseOwner() {
return houseOwner;
}
public void setHouseOwner(HouseOwner houseOwner) {
this.houseOwner = houseOwner;
}
public Tenant getTenant() {
return tenant;
}
public void setTenant(Tenant tenant) {
this.tenant = tenant;
}
public void constact(String message, Person person) {
// 如果是房主進行聯絡,則對應的租房者獲得房主的資訊
if (person == houseOwner) {
tenant.getMessage(message);
} else { //反之則是房主獲得資訊
houseOwner.getMessage(message);
}
}
}
//測試類
public class Client {
public static void main(String[] args) {
//一個房主、一個租房者、一箇中介機構
MediatorStructure mediator = new MediatorStructure();
//房主和租房者只需要知道中介機構即可
HouseOwner houseOwner = new HouseOwner("張三", mediator);
Tenant tenant = new Tenant("李四", mediator);
//中介結構要知道房主和租房者
mediator.setHouseOwner(houseOwner);
mediator.setTenant(tenant);
tenant.constact("需要租三室的房子");
houseOwner.constact("我這有三室的房子,你需要租嗎?");
}
}
首先我們給出中介者模式的適用場景:
然後我們給出中介者模式的優點:
鬆散耦合
中介者模式通過把多個同事物件之間的互動封裝到中介者物件裡面,從而使得同事物件之間鬆散耦合,基本上可以做到互補依賴。這樣一來,同事物件就可以獨立地變化和複用,而不再像以前那樣「牽一處而動全身」了。
集中控制互動
多個同事物件的互動,被封裝在中介者物件裡面集中管理,使得這些互動行為發生變化的時候,只需要修改中介者物件就可以了,當然如果是已經做好的系統,那麼就擴充套件中介者物件,而各個同事類不需要做修改。
一對多關聯轉變為一對一的關聯
沒有使用中介者模式的時候,同事物件之間的關係通常是一對多的,引入中介者物件以後,中介者物件和同事物件的關係通常變成雙向的一對一,這會讓物件的關係更容易理解和實現。
最後我們給出中介者模式的缺點:
下面我們來介紹迭代器模式
首先我們來簡單介紹一下迭代器模式:
迭代器模式主要包含以下角色:
抽象聚合(Aggregate)角色:定義儲存、新增、刪除聚合元素以及建立迭代器物件的介面。
具體聚合(ConcreteAggregate)角色:實現抽象聚合類,返回一個具體迭代器的範例。
抽象迭代器(Iterator)角色:定義存取和遍歷聚合元素的介面,通常包含 hasNext()、next() 等方法。
具體迭代器(Concretelterator)角色:實現抽象迭代器介面中所定義的方法,完成對聚合物件的遍歷,記錄遍歷的當前位置。
我們通過一個簡單的案例來介紹迭代器:
具體分析:
/*
【例】定義一個可以儲存學生物件的容器物件,將遍歷該容器的功能交由迭代器實現
Student:學生實體類
StudentIterator:抽象迭代器,宣告hasNext、next方法
StudentIteratorImpl:具體迭代器,重寫所有的抽象方法
StudentAggregate:抽象容器類,包含新增元素,刪除元素,獲取迭代器物件的方法
StudentAggregateImpl:具體的容器類,重寫所有的方法
*/
/* 程式碼展示 */
// 抽象迭代器
public interface StudentIterator {
boolean hasNext();
Student next();
}
// 具體迭代器
public class StudentIteratorImpl implements StudentIterator {
private List<Student> list;
private int position = 0;
public StudentIteratorImpl(List<Student> list) {
this.list = list;
}
@Override
public boolean hasNext() {
return position < list.size();
}
@Override
public Student next() {
Student currentStudent = list.get(position);
position ++;
return currentStudent;
}
}
// 抽象容器類
public interface StudentAggregate {
void addStudent(Student student);
void removeStudent(Student student);
StudentIterator getStudentIterator();
}
// 具體容器類(僅僅只是一個容器,用於生成一個統一的迭代器介面,實際內部還是呼叫我們上面定義的抽象迭代器的子類迭代器)
public class StudentAggregateImpl implements StudentAggregate {
private List<Student> list = new ArrayList<Student>(); // 學生列表
@Override
public void addStudent(Student student) {
this.list.add(student);
}
@Override
public void removeStudent(Student student) {
this.list.remove(student);
}
@Override
public StudentIterator getStudentIterator() {
return new StudentIteratorImpl(list);
}
}
首先我們給出迭代器模式的適用場景:
然後我們給出迭代器模式的優點:
最後我們給出迭代器模式的缺點:
下面我們來介紹存取者模式
首先我們來簡單介紹一下存取者模式:
存取者模式包含以下主要角色:
(Element)
存取的行為,它的引數就是可以存取的元素,它的方法個數理論上來講與元素類個數(Element的實現類個數)是一樣的,從這點不難看出,存取者模式要求元素類的個數不能改變。accept
),其意義是指,每一個元素都要可以被存取者存取。Element
),並且可以迭代這些元素,供存取者存取。我們給出一個簡單的案例來解釋存取者模式:
具體分析:
/*
【例】給寵物餵食
現在養寵物的人特別多,我們就以這個為例,當然寵物還分為狗,貓等,要給寵物餵食的話,主人可以喂,其他人也可以餵食。
- 存取者角色:給寵物餵食的人
- 具體存取者角色:主人、其他人
- 抽象元素角色:動物抽象類
- 具體元素角色:寵物狗、寵物貓
- 結構物件角色:主人家
*/
/* 程式碼展示 */
// 抽象存取者
public interface Person {
void feed(Cat cat);
void feed(Dog dog);
}
// 具體存取者(Owner主人,Someone其他人)
public class Owner implements Person {
@Override
public void feed(Cat cat) {
System.out.println("主人餵食貓");
}
@Override
public void feed(Dog dog) {
System.out.println("主人餵食狗");
}
}
public class Someone implements Person {
@Override
public void feed(Cat cat) {
System.out.println("其他人餵食貓");
}
@Override
public void feed(Dog dog) {
System.out.println("其他人餵食狗");
}
}
// 抽象節點
public interface Animal {
void accept(Person person);
}
// 具體節點
public class Dog implements Animal {
@Override
public void accept(Person person) {
person.feed(this);
System.out.println("好好吃,汪汪汪!!!");
}
}
public class Cat implements Animal {
@Override
public void accept(Person person) {
person.feed(this);
System.out.println("好好吃,喵喵喵!!!");
}
}
// 物件結構
public class Home {
// 需要操作的節點
private List<Animal> nodeList = new ArrayList<Animal>();
// 進行操作節點的存取者
public void action(Person person) {
for (Animal node : nodeList) {
// 存取者觸動了節點,以節點的形式呼叫方法
node.accept(person);
}
}
//新增操作
public void add(Animal animal) {
nodeList.add(animal);
}
}
// 測試類
public class Client {
public static void main(String[] args) {
Home home = new Home();
home.add(new Dog());
home.add(new Cat());
Owner owner = new Owner();
home.action(owner);
Someone someone = new Someone();
home.action(someone);
}
}
首先我們給出存取者模式的適用場景:
然後我們給出存取者模式的優點:
擴充套件性好
在不修改物件結構中的元素的情況下,為物件結構中的元素新增新的功能。
複用性好
通過存取者來定義整個物件結構通用的功能,從而提高複用程度。
分離無關行為
通過存取者來分離無關的行為,把相關的行為封裝在一起,構成一個存取者,這樣每一個存取者的功能都比較單一。
最後我們給出存取者模式的缺點:
物件結構變化很困難
在存取者模式中,每增加一個新的元素類,都要在每一個具體存取者類中增加相應的具體操作,這違背了「開閉原則」。
違反了依賴倒置原則
存取者模式依賴了具體類,而沒有依賴抽象類。
下面我們來介紹備忘錄模式
我們首先給出備忘錄模式的概念:
備忘錄模式的主要角色如下:
備忘錄有兩個等效的介面:
我們通過一個案例來介紹白箱備忘錄:
/*
【例】遊戲挑戰BOSS
遊戲中的某個場景,一遊戲角色有生命力、攻擊力、防禦力等資料,在打Boss前和後一定會不一樣的,我們允許玩家如果感覺與Boss決鬥的效果不理想可以讓遊戲恢復到決鬥之前的狀態。
【白箱備忘錄】
備忘錄角色對任何物件都提供一個介面,即寬介面,備忘錄角色的內部所儲存的狀態就對所有物件公開。
由於備忘錄角色對任何物件都提供寬介面(所有許可權),所以實際上是不安全的
*/
/* 程式碼展示 */
// 遊戲角色類(發起人)
public class GameRole {
private int vit; //生命力
private int atk; //攻擊力
private int def; //防禦力
//初始化狀態
public void initState() {
this.vit = 100;
this.atk = 100;
this.def = 100;
}
//戰鬥
public void fight() {
this.vit = 0;
this.atk = 0;
this.def = 0;
}
// 儲存角色狀態(建立出一個備忘錄,該備忘錄記錄當前狀況)
public RoleStateMemento saveState() {
return new RoleStateMemento(vit, atk, def);
}
// 回覆角色狀態(需要一個備忘錄,記錄需要恢復的狀態)
public void recoverState(RoleStateMemento roleStateMemento) {
this.vit = roleStateMemento.getVit();
this.atk = roleStateMemento.getAtk();
this.def = roleStateMemento.getDef();
}
public void stateDisplay() {
System.out.println("角色生命力:" + vit);
System.out.println("角色攻擊力:" + atk);
System.out.println("角色防禦力:" + def);
}
public int getVit() {
return vit;
}
public void setVit(int vit) {
this.vit = vit;
}
public int getAtk() {
return atk;
}
public void setAtk(int atk) {
this.atk = atk;
}
public int getDef() {
return def;
}
public void setDef(int def) {
this.def = def;
}
}
// 遊戲狀態儲存類(備忘錄類)
public class RoleStateMemento {
private int vit;
private int atk;
private int def;
public RoleStateMemento(int vit, int atk, int def) {
this.vit = vit;
this.atk = atk;
this.def = def;
}
public int getVit() {
return vit;
}
public void setVit(int vit) {
this.vit = vit;
}
public int getAtk() {
return atk;
}
public void setAtk(int atk) {
this.atk = atk;
}
public int getDef() {
return def;
}
public void setDef(int def) {
this.def = def;
}
}
// 角色狀態管理者類(管理者,內建一個備忘錄)
public class RoleStateCaretaker {
// 但是請注意,由於我們的備忘錄是一個類,我們的管理者可以對其roleStateMemento進行修改,因此是不符合要求的
private RoleStateMemento roleStateMemento;
public RoleStateMemento getRoleStateMemento() {
return roleStateMemento;
}
public void setRoleStateMemento(RoleStateMemento roleStateMemento) {
this.roleStateMemento = roleStateMemento;
}
}
// 測試類
public class Client {
public static void main(String[] args) {
System.out.println("------------大戰Boss前------------");
//大戰Boss前
GameRole gameRole = new GameRole();
gameRole.initState();
gameRole.stateDisplay();
//儲存進度
RoleStateCaretaker roleStateCaretaker = new RoleStateCaretaker();
roleStateCaretaker.setRoleStateMemento(gameRole.saveState());
System.out.println("------------大戰Boss後------------");
//大戰Boss時,損耗嚴重
gameRole.fight();
gameRole.stateDisplay();
System.out.println("------------恢復之前狀態------------");
//恢復之前狀態
gameRole.recoverState(roleStateCaretaker.getRoleStateMemento());
gameRole.stateDisplay();
}
}
我們通過一個案例來介紹黑箱備忘錄:
具體分析:
/*
【例】遊戲挑戰BOSS
遊戲中的某個場景,一遊戲角色有生命力、攻擊力、防禦力等資料,在打Boss前和後一定會不一樣的,我們允許玩家如果感覺與Boss決鬥的效果不理想可以讓遊戲恢復到決鬥之前的狀態。
【黑箱備忘錄】
備忘錄角色對發起人物件提供一個寬介面,而為其他物件提供一個窄介面。
在Java語言中,實現雙重介面的辦法就是將備忘錄類設計成發起人類的內部成員類。
*/
/* 程式碼展示 */
// 備忘錄 窄介面(只有一個標識,除發起人外都使用該介面,只有一個概念,無法對其進行操作)
public interface Memento {
}
// 遊戲角色類
public class GameRole {
// 前面的設定都是一樣的,但是在內部定義了備忘錄寬介面
private int vit; //生命力
private int atk; //攻擊力
private int def; //防禦力
//初始化狀態
public void initState() {
this.vit = 100;
this.atk = 100;
this.def = 100;
}
//戰鬥
public void fight() {
this.vit = 0;
this.atk = 0;
this.def = 0;
}
//儲存角色狀態
public Memento saveState() {
return new RoleStateMemento(vit, atk, def);
}
//回覆角色狀態
public void recoverState(Memento memento) {
RoleStateMemento roleStateMemento = (RoleStateMemento) memento;
this.vit = roleStateMemento.getVit();
this.atk = roleStateMemento.getAtk();
this.def = roleStateMemento.getDef();
}
public void stateDisplay() {
System.out.println("角色生命力:" + vit);
System.out.println("角色攻擊力:" + atk);
System.out.println("角色防禦力:" + def);
}
public int getVit() {
return vit;
}
public void setVit(int vit) {
this.vit = vit;
}
public int getAtk() {
return atk;
}
public void setAtk(int atk) {
this.atk = atk;
}
public int getDef() {
return def;
}
public void setDef(int def) {
this.def = def;
}
// 備忘錄寬介面,基於備忘錄窄介面的擴充套件,由於屬於發起人的內部類,發起人可以對其進行操作,備忘錄也可以獲得對應值
private class RoleStateMemento implements Memento {
private int vit;
private int atk;
private int def;
public RoleStateMemento(int vit, int atk, int def) {
this.vit = vit;
this.atk = atk;
this.def = def;
}
public int getVit() {
return vit;
}
public void setVit(int vit) {
this.vit = vit;
}
public int getAtk() {
return atk;
}
public void setAtk(int atk) {
this.atk = atk;
}
public int getDef() {
return def;
}
public void setDef(int def) {
this.def = def;
}
}
}
// 角色狀態管理者類(由於只有一個窄介面,所以無法進行操作,安全!)
public class RoleStateCaretaker {
private Memento memento;
public Memento getMemento() {
return memento;
}
public void setMemento(Memento memento) {
this.memento = memento;
}
}
// 使用者端
public class Client {
public static void main(String[] args) {
System.out.println("------------大戰Boss前------------");
//大戰Boss前
GameRole gameRole = new GameRole();
gameRole.initState();
gameRole.stateDisplay();
//儲存進度
RoleStateCaretaker roleStateCaretaker = new RoleStateCaretaker();
roleStateCaretaker.setMemento(gameRole.saveState());
System.out.println("------------大戰Boss後------------");
//大戰Boss時,損耗嚴重
gameRole.fight();
gameRole.stateDisplay();
System.out.println("------------恢復之前狀態------------");
//恢復之前狀態
gameRole.recoverState(roleStateCaretaker.getMemento());
gameRole.stateDisplay();
}
}
首先我們給出備忘錄模式的適用場景:
然後我們給出備忘錄模式的優點:
最後我們給出備忘錄模式的缺點:
最後我們介紹一下直譯器模式
我們首先給出直譯器模式的概念:
我們再來解釋一下文法:
我們給出一個文法案例:
expression ::= value | plus | minus
plus ::= expression ‘+’ expression
minus ::= expression ‘-’ expression
value ::= integer
// 注意:這裡的符號「::=」表示「定義為」的意思,豎線 | 表示或,左右的其中一個,引號內為字元本身,引號外為語法。
// 上面規則描述為 :
// 表示式可以是一個值,也可以是plus或者minus運算,而plus和minus又是由表示式結合運運算元構成,值的型別為整型數。
// 該內容比較生澀難懂,可以看一下案例會好懂很多~
直譯器模式包含以下主要角色。
抽象表示式(Abstract Expression)角色:定義直譯器的介面,約定直譯器的解釋操作,主要包含解釋方法 interpret()。
終結符表示式(Terminal Expression)角色:是抽象表示式的子類,用來實現文法中與終結符相關的操作,文法中的每一個終結符都有一個具體終結表示式與之相對應。
非終結符表示式(Nonterminal Expression)角色:也是抽象表示式的子類,用來實現文法中與非終結符相關的操作,文法中的每條規則都對應於一個非終結符表示式。
環境(Context)角色:通常包含各個直譯器需要的資料或是公共的功能,一般用來傳遞被所有直譯器共用的資料,後面的直譯器可以從這裡獲取這些值。
使用者端(Client):主要任務是將需要分析的句子或表示式轉換成使用直譯器物件描述的抽象語法樹,然後呼叫直譯器的解釋方法,當然也可以通過環境角色間接存取直譯器的解釋方法。
我們通過一個基本案例來理解直譯器模式:
具體分析:
/*
【例】設計實現加減法的軟體
首先我們來解釋幾個概念:
- 抽象表示式:書寫在運算過程中的所有Expression
- 終結表示式:不需要再計算,直接給出最終結果(Value,Variable)
- 非終結表示式:還不是最終結果,還需要繼續運算(Plus,Minus)
AbstractExpreesion:
核心點,明白來說就是Expreesion表示式,這裡的表示式包括有Variable變數,Value常數,也包括Plus和Minus運算
只有一個方法interpret,用於給子類去實現,如果是終結表示式直接返回結果,如果是非終結表示式則進一步運算,計算時均呼叫方法
*/
/* 程式碼展示 */
// 抽象角色AbstractExpression
public abstract class AbstractExpression {
// 只有一個方法,在運算時呼叫
public abstract int interpret(Context context);
}
// 終結符表示式角色(常數,例:1,2,3...)
public class Value extends AbstractExpression {
private int value;
public Value(int value) {
this.value = value;
}
// 直接返回結果即可
@Override
public int interpret(Context context) {
return value;
}
@Override
public String toString() {
return new Integer(value).toString();
}
}
// 非終結符表示式角色 加法表示式
public class Plus extends AbstractExpression {
// 既然是加法,那麼就有左右兩側,左右兩側同樣是Expression,既可以是終結符表示式,也可以是非終結符表示式
private AbstractExpression left;
private AbstractExpression right;
public Plus(AbstractExpression left, AbstractExpression right) {
this.left = left;
this.right = right;
}
// 運算時需要將左右兩側進行解釋並相加
@Override
public int interpret(Context context) {
return left.interpret(context) + right.interpret(context);
}
@Override
public String toString() {
return "(" + left.toString() + " + " + right.toString() + ")";
}
}
// 非終結符表示式角色 減法表示式
public class Minus extends AbstractExpression {
private AbstractExpression left;
private AbstractExpression right;
// 減法原理相同
public Minus(AbstractExpression left, AbstractExpression right) {
this.left = left;
this.right = right;
}
// 運算時需要將左右兩側進行解釋並相減
@Override
public int interpret(Context context) {
return left.interpret(context) - right.interpret(context);
}
@Override
public String toString() {
return "(" + left.toString() + " - " + right.toString() + ")";
}
}
// 終結符表示式角色 變數表示式(例:a=1,b=2...)
public class Variable extends AbstractExpression {
private String name;
public Variable(String name) {
this.name = name;
}
@Override
public int interpret(Context ctx) {
return ctx.getValue(this);
}
@Override
public String toString() {
return name;
}
}
// 環境類(存放變數的值)
public class Context {
private Map<Variable, Integer> map = new HashMap<Variable, Integer>();
public void assign(Variable var, Integer value) {
map.put(var, value);
}
public int getValue(Variable var) {
Integer value = map.get(var);
return value;
}
}
// 測試類
public class Client {
public static void main(String[] args) {
Context context = new Context();
Variable a = new Variable("a");
Variable b = new Variable("b");
Variable c = new Variable("c");
Variable d = new Variable("d");
Variable e = new Variable("e");
//Value v = new Value(1);
context.assign(a, 1);
context.assign(b, 2);
context.assign(c, 3);
context.assign(d, 4);
context.assign(e, 5);
AbstractExpression expression = new Minus(new Plus(new Plus(new Plus(a, b), c), d), e);
System.out.println(expression + "= " + expression.interpret(context));
}
}
我們首先給出直譯器模式的適用場景:
當語言的文法較為簡單,且執行效率不是關鍵問題時。
當問題重複出現,且可以用一種簡單的語言來進行表達時。
當一個語言需要解釋執行,並且語言中的句子可以表示為一個抽象語法樹的時候。
然後我們給出直譯器模式的優點:
易於改變和擴充套件文法。
由於在直譯器模式中使用類來表示語言的文法規則,因此可以通過繼承等機制來改變或擴充套件文法。每一條文法規則都可以表示為一個類,因此可以方便地實現一個簡單的語言。
實現文法較為容易。
在抽象語法樹中每一個表示式節點類的實現方式都是相似的,這些類的程式碼編寫都不會特別複雜。
增加新的解釋表示式較為方便。
如果使用者需要增加新的解釋表示式只需要對應增加一個新的終結符表示式或非終結符表示式類,原有表示式類程式碼無須修改,符合 "開閉原則"。
最後我們給出直譯器模式的缺點:
對於複雜文法難以維護。
在直譯器模式中,每一條規則至少需要定義一個類,因此如果一個語言套件含太多文法規則,類的個數將會急劇增加,導致系統難以管理和維護。
執行效率較低。
由於在直譯器模式中使用了大量的迴圈和遞迴呼叫,因此在解釋較為複雜的句子時其速度很慢,而且程式碼的偵錯過程也比較麻煩。
關於行為型模式就介紹到這裡了,目前我們已經學習了完整的二十三種設計模式,希望能為你帶來幫助~
該文章屬於學習內容,具體參考B站黑馬程式設計師的Java設計模式詳解
這裡附上視訊連結:10.設計模式-行為型模式概述_嗶哩嗶哩_bilibili