PHP設(shè)計(jì)模式之享元模式
享元模式,“享元”這兩個(gè)字在中文里其實(shí)并沒(méi)有什么特殊的意思,所以我們要把它拆分來(lái)看?!跋怼本褪枪蚕?,“元”就是元素,這樣一來(lái)似乎就很容易理解了,共享某些元素嘛。 Gof類(lèi)圖及解釋GoF定義:運(yùn)用共享技術(shù)有效地支持大量細(xì)粒度的對(duì)象 GoF類(lèi)圖
 代碼實(shí)現(xiàn)
interface Flyweight { public function operation($extrinsicState) : void; }
class ConcreteFlyweight implements Flyweight { private $intrinsicState = 101; function operation($extrinsicState) : void { echo '共享享元對(duì)象' . ($extrinsicState + $this->intrinsicState) . PHP_EOL; } }
class UnsharedConcreteFlyweight implements Flyweight { private $allState = 1000; public function operation($extrinsicState) : void { echo '非共享享元對(duì)象:' . ($extrinsicState + $this->allState) . PHP_EOL; } }
定義共享接口以及它的實(shí)現(xiàn),注意這里有兩個(gè)實(shí)現(xiàn),ConcreteFlyweigh進(jìn)行狀態(tài)的共享,UnsharedConcreteFlyweight不共享或者說(shuō)他的狀態(tài)不需要去共享 class FlyweightFactory { private $flyweights = [];
public function getFlyweight($key) : Flyweight { if (!array_key_exists($key, $this->flyweights)) { $this->flyweights[$key] = new ConcreteFlyweight(); } return $this->flyweights[$key]; } }
保存那些需要共享的對(duì)象,做為一個(gè)工廠(chǎng)來(lái)創(chuàng)建需要的共享對(duì)象,保證相同的鍵值下只會(huì)有唯一的對(duì)象,節(jié)省相同對(duì)象創(chuàng)建的開(kāi)銷(xiāo) $factory = new FlyweightFactory();
$extrinsicState = 100; $flA = $factory->getFlyweight('a'); $flA->operation(--$extrinsicState);
$flB = $factory->getFlyweight('b'); $flB->operation(--$extrinsicState);
$flC = $factory->getFlyweight('c'); $flC->operation(--$extrinsicState);
$flD = new UnsharedConcreteFlyweight(); $flD->operation(--$extrinsicState);
客戶(hù)端的調(diào)用,讓外部狀態(tài)$extrinsicState能夠在各個(gè)對(duì)象之間共享 有點(diǎn)意思吧,這個(gè)模式的代碼量可不算少 當(dāng)一個(gè)應(yīng)用程序使用了大量非常相似的的對(duì)象,對(duì)象的大多數(shù)狀都可變?yōu)橥獠繝顟B(tài)時(shí),很適合享元模式 這里的工廠(chǎng)是存儲(chǔ)對(duì)象列表的,不是像工廠(chǎng)方法或者抽象工廠(chǎng)一樣去創(chuàng)建對(duì)象的,雖說(shuō)這里也進(jìn)行了創(chuàng)建,但如果對(duì)象存在,則會(huì)直接返回,而且列表也是一直維護(hù)的 享元模式在現(xiàn)實(shí)中,大家多少一定用過(guò),各種池技術(shù)就是它的典型應(yīng)用:線(xiàn)程池、連接池等等,另外兩個(gè)一樣的字符串String類(lèi)型在php或Java中都是可以===的,這也運(yùn)用到了享元模式,它們連內(nèi)存地址都是一樣的,這不就是一種共享嘛 關(guān)于享元模式,有一個(gè)極其經(jīng)典的例子,比我下面的例子要好的多,那就是關(guān)于圍棋的棋盤(pán)。圍棋只有黑白兩色,所以?xún)蓚€(gè)對(duì)象就夠了,接下來(lái)呢?改變他們的位置狀態(tài)就好啦!有興趣的朋友可以搜搜哈! Laravel中的IoC容器可以看作是一種享元模式的實(shí)現(xiàn)。它把對(duì)象保存在數(shù)組中,在需要的時(shí)候通過(guò)閉包機(jī)制進(jìn)行取用,也有一些類(lèi)有共享一些狀態(tài)屬性的內(nèi)容。大家可以翻看代碼了解了解。
還是說(shuō)到科技以換殼為本這件事上。畢竟,大家都還是喜歡各種顏色的手機(jī)來(lái)彰顯自己的個(gè)性。之前說(shuō)過(guò),如果每種顏色我們都要做一條生產(chǎn)線(xiàn)的話(huà)那豈不是一項(xiàng)巨大的投入。還好,每個(gè)型號(hào)我們的工廠(chǎng)(享元工廠(chǎng))只生產(chǎn)最基本的背景殼(對(duì)象),然后通過(guò)專(zhuān)門(mén)的印刷線(xiàn)(狀態(tài)變化)來(lái)進(jìn)行上色不就好啦!嗯,下一款I(lǐng)phone早晚也會(huì)模仿我們的,看來(lái)我們得先把各種金、各種土豪色集齊才行,說(shuō)不定還能召喚神龍呢?。?/em> 完整代碼:https://github.com/zhangyue0503/designpatterns-php/blob/master/13.flyweights/source/flyweights.php 實(shí)例果然不出意外的我們還是來(lái)發(fā)短信,這回的短信依然使用的阿里云和極光短信來(lái)進(jìn)行發(fā)送,不過(guò)這次我們使用享元模式來(lái)實(shí)現(xiàn),這里的享元工廠(chǎng)我們保存了兩種不同類(lèi)型的對(duì)象哦,通過(guò)內(nèi)外狀態(tài)來(lái)讓它們千變?nèi)f化吧! 短信發(fā)送類(lèi)圖

完整源碼:https://github.com/zhangyue0503/designpatterns-php/blob/master/13.flyweights/source/flyweights-message.php <?php
interface Message { public function send(User $user); }
class AliYunMessage implements Message { private $template; public function __construct($template) { $this->template = $template; } public function send(User $user) { echo '使用阿里云短信向' . $user->GetName() . '發(fā)送:'; echo $this->template->GetTemplate(), PHP_EOL; } }
class JiGuangMessage implements Message { private $template; public function __construct($template) { $this->template = $template; } public function send(User $user) { echo '使用極光短信向' . $user->GetName() . '發(fā)送:'; echo $this->template->GetTemplate(), PHP_EOL; } }
class MessageFactory { private $messages = []; public function GetMessage(Template $template, $type = 'ali') { $key = md5($template->GetTemplate() . $type); if (!key_exists($key, $this->messages)) { if ($type == 'ali') { $this->messages[$key] = new AliYunMessage($template); } else { $this->messages[$key] = new JiGuangMessage($template); } } return $this->messages[$key]; }
public function GetMessageCount() { echo count($this->messages); } }
class User { public $name; public function GetName() { return $this->name; } }
class Template { public $template; public function GetTemplate() { return $this->template; } }
// 內(nèi)部狀態(tài) $t1 = new Template(); $t1->template = '模板1,不錯(cuò)喲!';
$t2 = new Template(); $t2->template = '模板2,還好啦!';
// 外部狀態(tài) $u1 = new User(); $u1->name = '張三';
$u2 = new User(); $u2->name = '李四';
$u3 = new User(); $u3->name = '王五';
$u4 = new User(); $u4->name = '趙六';
$u5 = new User(); $u5->name = '田七';
// 享元工廠(chǎng) $factory = new MessageFactory();
// 阿里云發(fā)送 $m1 = $factory->GetMessage($t1); $m1->send($u1);
$m2 = $factory->GetMessage($t1); $m2->send($u2);
echo $factory->GetMessageCount(), PHP_EOL; // 1
$m3 = $factory->GetMessage($t2); $m3->send($u2);
$m4 = $factory->GetMessage($t2); $m4->send($u3);
echo $factory->GetMessageCount(), PHP_EOL; // 2
$m5 = $factory->GetMessage($t1); $m5->send($u4);
$m6 = $factory->GetMessage($t2); $m6->send($u5);
echo $factory->GetMessageCount(), PHP_EOL; // 2
// 加入極光 $m1 = $factory->GetMessage($t1, 'jg'); $m1->send($u1);
$m2 = $factory->GetMessage($t1); $m2->send($u2);
echo $factory->GetMessageCount(), PHP_EOL; // 3
$m3 = $factory->GetMessage($t2); $m3->send($u2);
$m4 = $factory->GetMessage($t2, 'jg'); $m4->send($u3);
echo $factory->GetMessageCount(), PHP_EOL; // 4
$m5 = $factory->GetMessage($t1, 'jg'); $m5->send($u4);
$m6 = $factory->GetMessage($t2, 'jg'); $m6->send($u5);
echo $factory->GetMessageCount(), PHP_EOL; // 4
說(shuō)明
代碼有點(diǎn)多吧,但其實(shí)一共是兩種類(lèi)型的類(lèi),生成了四種對(duì)象。這里每個(gè)類(lèi)不同的對(duì)象是根據(jù)模板來(lái)區(qū)分的 這樣的組合還是比較方便的吧,再結(jié)合別的模式將工廠(chǎng)這里優(yōu)化一下,嗯,前途不可限量,你們可以想想哦! 享元模式適用于系統(tǒng)中存在大量的相似對(duì)象以及需要緩沖池的場(chǎng)景,能夠降低內(nèi)存占用,提高效率,但會(huì)增加復(fù)雜度,需要分享內(nèi)外部狀態(tài) 最主要的特點(diǎn)是有一個(gè)唯一標(biāo)識(shí),當(dāng)內(nèi)存中已經(jīng)有這個(gè)對(duì)象了,直接返回對(duì)象,不用再去創(chuàng)建了
下期看點(diǎn)享元模式的例子說(shuō)得有點(diǎn)牽強(qiáng),不過(guò)確實(shí)這類(lèi)設(shè)計(jì)模式在我們?nèi)粘5拈_(kāi)發(fā)中一方面用得不多,另一方面典型的例子又太經(jīng)典,反正只要記住它的特點(diǎn)就好了,具體在應(yīng)用的時(shí)候說(shuō)不準(zhǔn)寫(xiě)完代碼回頭一看你會(huì)發(fā)現(xiàn)這不就是我學(xué)過(guò)的享元模式嘛!好了,下一篇我們學(xué)習(xí)一個(gè)也是比較少用而且復(fù)雜的模式,但是也許你也能經(jīng)常見(jiàn)到哦!組合模式
|