站長資訊網
最全最豐富的資訊網站

PHP中什么是狀態模式?通過實例來了解它

在之前的文章《一起聊聊PHP中的單例模式》中我們介紹了PHP中的單例模式,下面本篇文章帶大家了解一下PHP設計模式中的狀態模式

PHP中什么是狀態模式?通過實例來了解它

狀態模式從字面上其實并不是很好理解。這里的狀態是什么意思呢?保存狀態?那不就是備忘錄模式了。其實,這里的狀態是類的狀態,通過改變類的某個狀態,讓這個類感覺像是換了一個類一樣。說起來有點拗口吧,先學習概念之后再看。

Gof類圖及解釋

GoF定義:允許一個對象在其內部狀態改變時改變它的行為。對象看起來似乎修改了它的類

GoF類圖

PHP中什么是狀態模式?通過實例來了解它

代碼實現

class Context {     private $state;     public function SetState(State $state): void     {         $this->state = $state;     }     public function Request(): void     {         $this->state = $this->state->Handle();     } }

一個上下文類,也可以看作是目標類,它的內部有一個狀態對象。當調用Request()的時候,去調用狀態類的Handle()方法。目的是當前上下文類狀態的變化都由外部的這個狀態類來進行操縱。

interface State {     public function Handle(): State; }  class ConcreteStateA implements State {     public function Handle(): State     {         echo '當前是A狀態', PHP_EOL;         return new ConcreteStateB();     } }  class ConcreteStateB implements State {     public function Handle(): State     {         echo '當前是B狀態', PHP_EOL;         return new ConcreteStateA();     } }

抽象狀態接口及兩個具體實現。這兩個具體實現實際上是在相互調用。實現的效果就是上下文類每調用一次Request()方法,內部的狀態類就變成別一個狀態。就像一個開關,在打開與關閉中來回切換一樣。

$c = new Context(); $stateA = new ConcreteStateA(); $c->SetState($stateA); $c->Request(); $c->Request(); $c->Request(); $c->Request();

客戶端的實現,實例化上下文對象并設置初始的狀態,然后通過不停的調用Request()對象來實現開關狀態的切換。

  • 看出門道了嘛?這里把狀態的變化給封裝到外部的實現類去了,并不是這個上下文或者目標類內部來進行狀態的切換了
  • 那么狀態模式的意義呢?這個默認類圖的例子過于簡單,其實狀態模式的真正目的是為了解決復雜的if嵌套問題的,把復雜的if嵌套條件放到一個個的外部狀態類中去判斷,在后面的實例中我們會看到
  • 適用于:一個對象的行為取決于它的狀態,并且它的必須在運行時刻根據狀態改變自己的行為;一個操作中含有大量的多分支條件語句,且這些分支依賴于該對象的狀態;
  • 狀態模式的特點是:它將與特定狀態相關的行為局部化;它使得狀態轉換顯式化;State對象可以被共享;
  • 常見于訂單系統、會員系統、OA系統中,也就是流程中會出現各種狀態變化的情況,都可以使用狀態模式來進行整體的設計與架構

我們的手機系統內定制了自己的商城系統,可以在手機上方便的下單購買我們的商品。一個訂單(Context)會有多種狀態(State),比如未支付、已支付、訂單完成、訂單退款等等一大堆狀態。我們把這些狀態都放在了對應的狀態類里去實現,不同的狀態類都會再去調用該狀態下一步的動作,比如已支付后就等待收貨、退款后就等待買家填寫物流單號等,這樣,狀態模式就在我們的商城中被靈活的運用起來咯!!

完整代碼:https://github.com/zhangyue0503/designpatterns-php/blob/master/22.state/source/state.php

實例

通常的商城應用中都會有會員體系的存在,一般等級越高的會員可以享受的折扣也會越多,這個時候,運用狀態模式就能很輕松的獲得會員的等級折扣。當然,最主要的是,使用狀態模式可以在需要添加或者刪除會員等級時只添加對應的會員折扣狀態子類就可以了。其他業務代碼都不需要變動,我們一起來看看具體實現吧!

會員折扣圖

PHP中什么是狀態模式?通過實例來了解它

完整源碼:https://github.com/zhangyue0503/designpatterns-php/blob/master/22.state/source/state-member.php

<?php  class Member {     private $state;     private $score;      public function SetState($state)     {         $this->state = $state;     }      public function SetScore($score)     {         $this->score = $score;     }      public function GetScore()     {         return $this->score;     }      public function discount()     {         return $this->state->discount($this);     } }  interface State {     public function discount($member); }  class PlatinumMemeberState implements State {     public function discount($member)     {         if ($member->GetScore() >= 1000) {             return 0.80;         } else {             $member->SetState(new GoldMemberState());             return $member->discount();         }     } }  class GoldMemberState implements State {     public function discount($member)     {         if ($member->GetScore() >= 800) {             return 0.85;         } else {             $member->SetState(new SilverMemberState());             return $member->discount();         }     } }  class SilverMemberState implements State {     public function discount($member)     {         if ($member->GetScore() >= 500) {             return 0.90;         } else {             $member->SetState(new GeneralMemberState());             return $member->discount();         }     } }  class GeneralMemberState implements State {     public function discount($member)     {         return 0.95;     } }  $m = new Member(); $m->SetState(new PlatinumMemeberState());  $m->SetScore(1200); echo '當前會員' . $m->GetScore() . '積分,折扣為:' . $m->discount(), PHP_EOL;  $m->SetScore(990); echo '當前會員' . $m->GetScore() . '積分,折扣為:' . $m->discount(), PHP_EOL;  $m->SetScore(660); echo '當前會員' . $m->GetScore() . '積分,折扣為:' . $m->discount(), PHP_EOL;  $m->SetScore(10); echo '當前會員' . $m->GetScore() . '積分,折扣為:' . $m->discount(), PHP_EOL;

說明

  • 如果不使用狀態模式,在Member的discount()方法中,我們可能需要寫很多層if…else…判斷條件
  • 同時,這也帶來了方法體會越來越長,越來越難以維護的問題
  • 狀態模式正是為了解決這個問題而存在的
  • 當discount()行為的結果依賴于Member對象本身的狀態(會員分)時,狀態模式就是最佳的選擇了,也就是上面所說的一個對象的行為取決于它的狀態

原文地址:https://juejin.cn/post/6844903991562731534

作者:硬核項目經理

推薦學習:《PHP視頻教程》

贊(0)
分享到: 更多 (0)
網站地圖   滬ICP備18035694號-2    滬公網安備31011702889846號
91精品国产综合久久香蕉| 亚洲欧美中文日韩视频| 九九热这里只有在线精品视| 手机在线看片国产日韩生活片| 2022国产精品视频| 日韩精品无码免费专区午夜不卡| 中文字幕日韩精品在线| 91麻豆国产精品91久久久| 亚洲AV永久无码精品一百度影院 | 午夜精品久久久久久影视777 | 国产精品 猎奇 另类视频| 国产精品福利在线观看| 成人日韩熟女高清视频一区| 国内精品综合久久久40p| 日韩高清在线中文字带字幕| 亚洲精品精华液一区二区| 亚洲国产精品婷婷久久| 精品无码国产自产拍在线观看蜜 | 日韩精品一区二区三区老鸦窝| 亚洲色图国产精品| 久久精品三级视频| 日韩精品国产另类专区| 99re最新在线精品| 人妻少妇精品中文字幕av蜜桃| 精品久久久久久国产三级| 在线观看亚洲AV日韩AV| 精品国产一区二区三区久久影院| 久久精品国产亚洲av麻豆图片 | 国产91久久精品一区二区| 久久国产精品一国产精品| 国产精品女人呻吟在线观看| 亚洲AV日韩AV鸥美在线观看| 国产乱码精品一区二区三区 | 日韩中文字幕精品免费一区| 亚洲AV永久无码精品成人| 国产精品嫩草影院一二三区| 蜜臀色欲AV在线播放国产日韩| 日产精品卡2卡三卡乱码网址| 国产精品视频白浆合集| 2019国产精品| 久久久久久久久久国产精品免费|