java設計模型


設計模式分為三種類型,共23種:

  • 創建型模式:單例模式、抽象工廠模式、建造者模式、工廠模式、原型模式。
  • 結構型模式:適配器模式、橋接模式、裝飾模式、組合模式、外觀模式、享元模式、代理模式。
  • 行為型模式:模版方法模式、命令模式、迭代器模式、觀察者模式、中介者模式、備忘錄模式、解釋器模式、狀態模式、策略模式、職責鏈模式(責任鏈模式)、訪問者模式。

模式分析:

1.觀察者模式:


  定義了對象之間的一對多的依賴,這樣一來,當一個對象改變時,它的所有的依賴者都會收到         通知並自動更新。

  • 對於JDK或者Andorid中都有很多地方實現了觀察者模式,比如XXXView.addXXXListenter , 當然了 XXXView.setOnXXXListener不一定是觀察者模式,因為觀察者模式是一種一對多的關系,對於setXXXListener是1對1的關系,應該叫回調。
  • 專題接口:Subject.java ;

  1. /**
  2. * 注冊一個觀察者
  3. */
  4. public void registerObserver(Observer observer);
  5. /**
  6. * 移除一個觀察者
  7. */
  8. public void removeObserver(Observer observer);
  9. /**
  10. * 通知所有觀察者
  11. */
  12. public void notifyObservers();
  • 3D服務號的實現類:ObjectFor3D.java

  1. @Override
  2. public void registerObserver(Observer observer) {
  3. observers.add(observer);
  4. }
  5. @Override
  6. public void removeObserver(Observer observer) {
  7. int index = observers.indexOf(observer);
  8. if (index >= 0) {
  9. observers.remove(index);
  10. }
  11. }
  12. @Override
  13. public void notifyObservers() {
  14. for (Observer observer : observers) {
  15. observer.update(msg);
  16. }
  17. }
  18. /**
  19. * 主題更新信息
  20. */
  21. public void setMsg(String msg) {
  22. this.msg = msg;
  23. notifyObservers();
  24. }
  • 所有觀察者需要實現此接口:Observer.java

  1. public ObserverUser1(Subject subject) {
  2. subject.registerObserver(this);
  3. }
  4. @Override
  5. public void update(String msg) {
  6. Log.e("-----ObserverUser1 ", "得到 3D 號碼:" + msg + ", 我要記下來。 ");
  7. }
  • 最后測試:ObserverActivity.java

  1. // 創建服務號
  2. objectFor3D = new ObjectFor3D();
  3. // 創建兩個訂閱者
  4. observerUser1 = new ObserverUser1(objectFor3D);
  5. observerUser2 = new ObserverUser2(objectFor3D);
  6. // 兩個觀察者,發送兩條信息
  7. objectFor3D.setMsg("201610121 的3D號為:127");
  8. objectFor3D.setMsg("20161022 的3D號為:000");

2. 工廠模式:


簡單列一下這個模式的家族:

1、靜態工廠模式

  • 這個最常見了,項目中的輔助類,TextUtil.isEmpty等,類+靜態方法。

2、簡單工廠模式(店里買肉夾饃)

  • 定義:通過專門定義一個類來負責創建其他類的實例,被創建的實例通常都具有共同的父類。
  • 根據類型直接創建肉夾饃:SimpleRoujiaMoFactory.java
  1. public RoujiaMo creatRoujiaMo(String type) {
  2. RoujiaMo roujiaMo = null;
  3. switch (type) {
  4. case "Suan":
  5. roujiaMo = new ZSuanRoujiaMo();
  6. break;
  7. case "La":
  8. roujiaMo = new ZLaRoujiaMo();
  9. break;
  10. case "Tian":
  11. roujiaMo = new ZTianRoujiaMo();
  12. break;
  13. default:// 默認為酸肉夾饃
  14. roujiaMo = new ZSuanRoujiaMo();
  15. break;
  16. }
  17. return roujiaMo;
  18. }

3、工廠方法模式(開分店)

  • 定義:定義一個創建對象的接口,但由子類決定要實例化的類是哪一個。工廠方法模式把類實例化的過程推遲到子類。
  • 對比定義:
    • 1、定義了創建對象的一個接口:public abstract RouJiaMo sellRoujiaMo(String type);
    • 2、由子類決定實例化的類,可以看到我們的饃是子類生成的。
    • 提供創建肉夾饃店抽象方法:RoujiaMoStore.java
    1.   public abstract RoujiaMo sellRoujiaMo(String type);
    • 具體實現抽象方法:XianRoujiaMoStore.java
    • 分店依舊使用簡單工廠模式:XianSimpleRoujiaMoFactory.java

4、抽象工廠模式(使用官方提供的原料)

  • 定義:提供一個接口,用於創建相關的或依賴對象的家族,而不需要明確指定具體類。
  • 對比定義:
    • 1、提供一個接口:public interface RouJiaMoYLFactroy
    • 2、用於創建相關的或依賴對象的家族 public Meat createMeat();public YuanLiao createYuanliao();我們接口用於創建一系列的原材料。
  • 創建用於提供原料的接口工廠:RoujiaMoYLFactory.java
  • 各自分店實現接口,完成原料提供:XianRoujiaMoYLFoctory.java
  • 准備時,使用官方的原料:RoujiaMo.java
  1. /**
  2. * 准備工作
  3. */
  4. public void prepare(RoujiaMoYLFactory roujiaMoYLFactory) {
  5. Meet meet = roujiaMoYLFactory.creatMeet();
  6. YuanLiao yuanLiao = roujiaMoYLFactory.creatYuanLiao();
  7. Log.e("---RoujiaMo:", "使用官方的原料 ---" + name + ": 揉面-剁肉-完成准備工作 yuanLiao:"+meet+"yuanLiao:"+yuanLiao);
  8. }

3. 單例設計模式:


單例模式主要是為了避免因為創建了多個實例造成資源的浪費,且多個實例由於多次調用容易導致結果出現錯誤,而使用單例模式能夠保證整個應用中有且只有一個實例。

  • 定義:只需要三步就可以保證對象的唯一性
    • (1) 不允許其他程序用new對象
    • (2) 在該類中創建對象
    • (3) 對外提供一個可以讓其他程序獲取該對象的方法
  • 對比定義:
    • (1) 私有化該類的構造函數
    • (2) 通過new在本類中創建一個本類對象
    • (3) 定義一個公有的方法,將在該類中所創建的對象返回
  • 餓漢式[可用]:SingletonEHan.java
  • 含懶漢式[雙重校驗鎖 推薦用]:SingletonLanHan.java
  1. private SingletonLanHan() {}
  2. private static SingletonLanHan singletonLanHanFour;
  3. public static SingletonLanHan getSingletonLanHanFour() {
  4. if (singletonLanHanFour == null) {
  5. synchronized (SingletonLanHan.class) {
  6. if (singletonLanHanFour == null) {
  7. singletonLanHanFour = new SingletonLanHan();
  8. }
  9. }
  10. }
  11. return singletonLanHanFour;
  12. }
  • 內部類[推薦用]:SingletonIn.java
  • 枚舉[推薦用]:SingletonEnum.java

 

4. 策略模式:


策略模式:定義了算法族,分別封裝起來,讓它們之間可相互替換,此模式讓算法的變化獨立於使用算法的客戶。

  • 以創建游戲角色為例子:
    • 最初的游戲角色的父類:Role.java
    • 發現有重復代碼后,重構后的父類:Role.java
  • 總結:
    • 1、封裝變化(把可能變化的代碼封裝起來)
    • 2、多用組合,少用繼承(我們使用組合的方式,為客戶設置了算法)
    • 3、針對接口編程,不針對實現(對於Role類的設計完全的針對角色,和技能的實現沒有關系)
  • 最后測試:創建角色:
  1. RoleA roleA = new RoleA("---A");
  2. roleA.setiDisplayBehavior(new DisplayYZ())
  3. .setiAttackBehavior(new AttackXL())
  4. .setiDefendBehavior(new DefendTMS())
  5. .setiRunBehavior(new RunJCTQ());
  6. roleA.display();// 樣子
  7. roleA.attack();// 攻擊
  8. roleA.run();// 逃跑
  9. roleA.defend();// 防御

5. 適配器模式:


定義:將一個類的接口轉換成客戶期望的另一個接口,適配器讓原本接口不兼容的類可以相互合作。這個定義還好,說適配器的功能就是把一個接口轉成另一個接口。

  • 以充電器為實例: 手機充電器一般都是5V左右吧,咱天朝的家用交流電壓220V,所以手機充電需要一個適配器(降壓器)
  • 一部手機: Mobile.java
  • 手機依賴一個提供5V電壓的接口: V5Power.java
  • 我們擁有的是220V家用交流電: V220Power.java
  • 適配器,完成220V轉5V的作用:V5PowerAdapter.java
  • 最后測試:給手機沖個電:
  1. Mobile mobile = new Mobile();
  2. V5Power v5Power = new V5PowerAdapter(new V200Power());
  3. mobile.inputPower(v5Power);

6. 命令模式:


定義:將“請求”封裝成對象,以便使用不同的請求、隊列或者日志來參數化其他對象。命令模式也支持可撤銷的操作。(簡化: 將請求封裝成對象,將動作請求者和動作執行者解耦。)

  • 需求:最近智能家電很火熱,假設現在有電視、電腦、電燈等家電,現在需要你做個遙控器控制所有家電的開關,要求做到每個按鈕對應的功能供用戶個性化,對於新買入家電要有非常強的擴展性。
  • 1、家電的API:Door.java
  • 2、把命令封裝成類:
    • 統一的命令接口:Command.java
    • 家電實現該接口:DoorOpenCommand.java
  • 3、遙控器:ControlPanel.java
  • 4、定義一個命令,可以干一系列的事情:QuickCommand.java
  1. QuickCommand quickCloseCommand = new QuickCommand(new Command[]{new LightOffCommand(light), new ComputerOffCommand(computer), new DoorCloseCommand(door)});
  2. controlPanel.setCommands(6, quickOpenCommand);
  3. controlPanel.keyPressed(6);
  • 5、遙控器面板執行:CommandActivity.java

  1. controlPanel.setCommands(0, new DoorOpenCommand(door));// 開門
  2. controlPanel.keyPressed(0);

7. 裝飾者模式:


裝飾者模式:若要擴展功能,裝飾者提供了比集成更有彈性的替代方案,動態地將責任附加到對象上。

  • 先簡單描述下裝飾者模式發揮作用的地方,當我們設計好了一個類,我們需要給這個類添加一些輔助的功能,並且不希望改變這個類的代碼,這時候就是裝飾者模式大展雄威的時候了。這里還體現了一個原則:類應該對擴展開放,對修改關閉。
  • 需求:設計游戲的裝備系統,基本要求,要可以計算出每種裝備在鑲嵌了各種寶石后的攻擊力和描述:
  • 1、裝備的超類:IEquip.java
  • 2、各個裝備的實現類:
    • eg:武器的實現類: ArmEquip.java
  • 3、裝飾品的超類(裝飾品也屬於裝備):IEquipDecorator.java
  • 4、裝飾品的實現類:
    • eg:藍寶石的實現類(可累加): BlueGemDecorator.java
  • 5、最后測試:計算攻擊力和查看描述:
  1. Log.e("---", "一個鑲嵌2顆紅寶石,1顆藍寶石的靴子: ");
  2. IEquip iEquip = new RedGemDecotator(new RedGemDecotator(new BlueGemDecotator(new ShoeEquip())));
  3. Log.e("---", "攻擊力:" + iEquip.caculateAttack());
  4. Log.e("---", "描述語:" + iEquip.description());

8. 外觀模式:


定義:提供一個統一的接口,用來訪問子系統中的一群接口,外觀定義了一個高層的接口,讓子系統更容易使用。其實就是為了方便客戶的使用,把一群操作,封裝成一個方法。

  • 需求:我比較喜歡看電影,於是買了投影儀、電腦、音響、設計了房間的燈光、買了爆米花機,然后我想看電影的時候,我需要一鍵觀影和一鍵關閉。
  • 每個設備類的開關等操作:
    • eg: 爆米花機:PopcornPopper.java
  • 電影院類:HomeTheaterFacade.java
  1. /**
  2. * 一鍵觀影
  3. */
  4. public void watchMovie() {
  5. computer.on();
  6. light.down();
  7. popcornPopper.on();
  8. popcornPopper.makePopcorn();
  9. projector.on();
  10. projector.open();
  11. player.on();
  12. player.make3DListener();
  13. }
  • 最后測試:一鍵觀影:
  1. new HomeTheaterFacade(computer, light, player, popcornPopper, projector).watchMovie();

9. 模板方法模式:


定義:定義了一個算法的骨架,而將一些步驟延遲到子類中,模版方法使得子類可以在不改變算法結構的情況下,重新定義算法的步驟。

  • 需求:簡單描述一下:本公司有程序猿、測試、HR、項目經理等人,下面使用模版方法模式,記錄下所有人員的上班情況
  • 模板方法模式中的三類角色
    • 1、具體方法(Concrete Method)
    • 2、抽象方法(Abstract Method)
    • 3、鈎子方法(Hook Method)
  • 工人的超類:Worker.java
  1. // 具體方法
  2. public final void workOneDay() {
  3. Log.e("workOneDay", "-----------------work start----------------");
  4. enterCompany();
  5. work();
  6. exitCompany();
  7. Log.e("workOneDay", "-----------------work end----------------");
  8. }
  9. // 工作 抽象方法
  10. public abstract void work();
  11. // 鈎子方法
  12. public boolean isNeedPrintDate() {
  13. return false;
  14. }
  15. private void exitCompany() {
  16. if (isNeedPrintDate()) {
  17. Log.e("exitCompany", "---" + new Date().toLocaleString() + "--->");
  18. }
  19. Log.e("exitCompany", name + "---離開公司");
  20. }
  • 程序員實現類(可得知時間):ITWorker.java

  1. /**
  2. * 重寫父類的此方法,使可以查看離開公司時間
  3. */
  4. @Override
  5. public boolean isNeedPrintDate() {
  6. return true;
  7. }
  • 最后測試:
    • 查看所有人員的工作情況:
  1. QAWorker qaWorker = new QAWorker("測試人員");
  2. qaWorker();
  3. HRWorker hrWorker = new HRWorker("莉莉姐");
  4. hrWorker.workOneDay();
  5. ...
    • 查看程序猿離開公司的時間:
  1. ITWorker itWorker = new ITWorker("jingbin");
  2. itWorker.workOneDay();

10. 狀態模式:


定義:允許對象在內部狀態改變時改變它的行為,對象看起來好像修改了它的類。

  • 定義又開始模糊了,理一下,當對象的內部狀態改變時,它的行為跟隨狀態的改變而改變了,看起來好像重新初始化了一個類似的。
  • 需求:已自動售貨機為例(有已投幣、未投幣等狀態和投幣、投幣等方法)
  • 最初實現待改進的售貨機:VendingMachine.java
  • 改進后的售貨機(更具有延展性):VendingMachineBetter.java
  1. // 放錢
  2. public void insertMoney() {
  3. currentState.insertMoney();
  4. }
  5. // 退錢
  6. public void backMoney() {
  7. currentState.backMoney();
  8. }
  9. // 轉動曲柄
  10. public void turnCrank() {
  11. currentState.turnCrank();
  12. if (currentState == soldState || currentState == winnerState) {
  13. currentState.dispense();//兩種情況會出貨
  14. }
  15. }
  16. // 出商品
  17. public void dispense() {
  18. Log.e("VendingMachineBetter", "---發出一件商品");
  19. if (count > 0) {
  20. count--;
  21. }
  22. }
  23. // 設置對應狀態
  24. public void setState(State state) {
  25. this.currentState = state;
  26. }
  • 狀態的接口:State.java

  • 對應狀態的接口實現類:
    • eg: 中獎狀態:WinnerState.java
    • eg: 售賣狀態:SoldState.java
  • 改進后的售貨機測試:
  1. // 初始化售貨機,且里面有3個商品
  2. VendingMachineBetter machineBetter = new VendingMachineBetter(3);
  3. machineBetter.insertMoney();
  4. machineBetter.turnCrank();

參考鏈接

https://blog.csdn.net/GarfieldEr007/article/details/78704562


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM