初學者對於設計模式肯定存在著很多不明白之處,今天剛好週末,就抽出來點時間寫了一個單件模式結合命令鏈模式打造系統核心的文章,可能對於部分人來說,文章內容過於淺顯,這是送給初學者的教程,因為時間比較緊(要陪老婆逛街,呵呵),其中出現了設計不規範的,代碼書寫不規範的,bug等等還望各路大俠指出來,方便大家共同進步.本人水平有限.^_^
相信大家都已經讀過很多關於在php中應用設計模式的書籍或是文章,但是很少有直接給予實例,大部分看完之後有種迷迷糊糊的感覺,如果沒有專案實踐,很難將設計模式部分弄清楚.
為避免程式碼過於複雜.沒有添加異常處理等內容.
單件模式以及命令鏈模式的基礎知識,大家自己google一下.不詳細講了.下面直接看實例:
<?php
/*
*@author:NoAngels
*@time:08年08月30日
*/
interface IRunAction{
//取得類別中定義的可以被APP中run的方法
static function LoadActions();
//類別中的入口函數呼叫該類別中其他函數用
function runAction($action, $args);
}
/*
*APP類系統的核心部分
*/
class APP{
static private $__instance = null;
static private $__commands = array();
static private $__flag = 1;
private function __construct(){}
//單件模式設計取得該類別的唯一實例
static function Load(){
if(self::$__instance == null) self::$__instance = new APP;
return self::$__instance;
}
//新增命名到APP的$__instance中每次新增指令的時候檢查是否之前已經新增過一個該類別的實例
//如果有就忽略操作如果沒有就加入進來
public function addCommand($cmdName){
foreach(self::$__commands as $cmd){
if(strtolower(get_class($cmd)) == strtolower(get_class($cmdName))){
self::$__flag = 0;
break;
}
}
if(self::$__flag == 1) self::$__commands[] = $cmdName;
self::$__flag = 1;
}
//命令鏈模式設計的核心部分呼叫實例的入口函數
//先檢查是否在類別中允許呼叫該操作如果沒有就提示未定義操作退出
public function runCommand($action, $args){
self::$__flag = 0;
foreach(self::$__commands as $cmd){
if(in_array($action, $cmd->LoadActions())){
self::$__flag = 1;
$cmd->runAction($action, $args);
}
}
if(self::$__flag == 0){
self::$__flag = 1;
exit("undefined action by action : $action");
}
}
//刪除某個類別的實例,只要指定類別的名字即可
public function removeCommand($className){
foreach(self::$__commands as $key=>$cmd){
if(strtolower(get_class($cmd)) == strtolower($className)){
unset(self::$__commands[$key]);
}
}
}
//供大家測試用看看是否新增以及刪除成功
public function viewCommands(){
echo(count(self::$__commands));
}
}
//類別User實作介面IRunAction
class User implements IRunAction{
//定義可以呼叫的操作
static private $__actions = array('addUser', 'modifyUser', 'removeUser');
//取得可以調用的操作,實際過程中不要直接就愛你個$__actions設計成public調用
//而應該設計一個LoadActions函數來取得$__actions的值
static public function LoadActions(){
return self::$__actions;
}
//執行指定函數
public function runAction($action, $args){
//不明白這個函數使用的可以參考手冊
call_user_func(array($this,$action), $args);
}
//測試函數而已
protected function addUser($name){
echo($name);
}
}
//類別Test同類User
class Test implements IRunAction{
static private $__actions = array('addTest', 'modifyTest', 'removeTest');
static public function LoadActions(){
return self::$__actions;
}
public function runAction($action, $args){
call_user_func(array($this,$action), $args);
}
protected function addTest($name){
echo($name);
}
}
//以下是測試程式碼
APP::Load()->addCommand(new User);
APP::Load()->addCommand(new User);
APP::Load()->addCommand(new User);
APP::Load()->addCommand(new User);
APP::Load()->runCommand('addUser', 'NoAngels');
APP::Load()->addCommand(new Test);
APP::Load()->runCommand('addTest', null);
APP類別用單件模式設計,它是系統的核心部分.相信大家看程式碼就知道了Load方法是載入APP類別實例,相當於有些書中的getInstance 靜態方法.他有addCommand,runCommand,removeCommand三個public方法.runCommand是核心部分.同時也是命令鏈模式的核心啟動程序.具體實現請看源代碼.代碼寫的已經很清楚了,就此不再贅述.
類別User,Test實現了介面IRunAction,這兩個類別中都定義了一個靜態私有變數$__actions,為一數組,其中包含了可以被APP的runCommand函數呼叫的操作.
下面是系統的運行流程:
APP啟動
-------addCommand,將要運行的操作所屬的類別加入APP中.如果新增的類別是用單件模式設計的.可以如下新增addCommand(SingletonClass::Load()).否則可以如下調
addCommand(new someClass)
-------runCommand.執行操作.例如在User類別中有一操作addUser.我直接可以啟用runCommand($acttion, $args).在APP中循環遍歷$__commands數組,如果其中某個類別的實例擁有該操作,就呼叫該實例的runAction函數.如果你沒有將某個類別的實例利用addCommand加入進來,就提示未定義操作,退出.
在類別User和類別Test中的runAction調用了call_user_func這個非常使用的函數.調用該類別中對應函數.
提示:講解與實例部分就到此了,具體你怎麼理解,以及如何將該思想利用,就看你自己的理解,凡事必須自己動手才行.(ps:可以做成框架中的單一入口文件,實不實現MVC就看你自己是怎麼想的了.)