PHP面向對象之選擇工廠和更新工廠

来源:http://www.cnblogs.com/kerryw/archive/2017/06/29/7096904.html
-Advertisement-
Play Games

/* 選擇工廠和更新工廠模式,這個模式的類(UpdateFactory和SelectionFactory類)就是用來創建SQL語句的. 因為涉及到之前學習的內容比較多,這裡就儘量將之前相關模式的示例代碼放在一起來進行學習和回顧了。 以下的代碼都是代碼片段而且涉及到連接資料庫,無法進行整體的調試(某些... ...


/*
選擇工廠和更新工廠模式,這個模式的類(UpdateFactorySelectionFactory類)就是用來創建SQL語句的.
因為涉及到之前學習的內容比較多,這裡就儘量將之前相關模式的示例代碼放在一起來進行學習和回顧了。
以下的代碼都是代碼片段而且涉及到連接資料庫,無法進行整體的調試(某些部分單獨拿出來的話就可以),因此重在理解。
*/



//更新工廠
abstract class UpdateFactory{
    
    abstract function newUpdate(\woo\domain\DomainObject $obj);
    
    //拼接更新或插入的sql語句
    protected function buildStatement($table,array $fields ,array $condition = null){
        $terms = array(); //SQL語句中占位符所代表的實際值
        if(!is_null($condition)){        //有條件則拼接更新語句,無條件則拼接插入語句
            $query = "UPDATE {$table} SET";
            $query .= implode(" = ? " ,array_keys($fields)) . " = ? ";
            $terms = array_values($fields);
            $cond = array();
            $query .= " WHERE ";
            foreach($condition as $key=>$val){
                $cond[] = " $key = ? ";
                $terms[] = $val;
            }
            $query .= implode(" AND ",$cond);
        } else {
            $query = " INSERT INTO {$table} (";
            $query .= implode(",",array_keys($fields));
            $query .= " ) VALUES ( ";
            foreach($fields as $name => $value){
                $terms[] = $value;
                $qs[] = "?";
            }
            $query .= implode(",",$qs);
            $query .=" ) ";
        }
        return array($query,$terms);
    }
}

//通過領域模型生成SQL語句
class VenueUpdateFactory extends UpdateFactory{
    function newUpdate(\woo\domain\DomainObject $obj){
        $id = $obj->getId();
        $cond = null;
        $values['name'] = $obj->getName();
        if($id > -1 ){
            $cond["id"] = $id;
        }
        return $this->buildStatement("venue",$values,$cond);
    }
}


//選擇工廠
abstract class SelectionFactory{
    abstract function newSelection(IdentityObject $obj);
    
    //通過標識對象拼接sql語句的where條件語句
    function buildWhere (IdentityObject $obj){
        if($obj->isVoid){
            return array("",array());
        }
        $compstrings = array();
        $values = array();
        foreach($obj->getComps() as $comp){
            $compstrings[] = "{$comp['name']} {$comp['operator']} ?";
            $values[] = $comp['value'];
        }
        $where =  " WHERE " . implode(" AND ",$compstrings);
        return array($where,$values);
    }
}

//拼接select語句
class VenuSelectionFactory extends SelectionFactory {
    function newSelection(IdentityObject $obj){
        $field = implode(',',$obj->getObjectFields());
        $core = "SELECT $fields FROM venue";
        list($where,$values) = $this->buildWhere($obj);
        return array($core." ".$where,$values);
    }
}

//現在來回顧一下之前學習過的相關知識
//欄位對象
class Field {
    protected $name = null;          //欄位名稱
    protected $operator = null;         //操作符    
    protected $comps = array();         //存放條件的數組    
    protected $incomplete = false;     //檢查條件數組是否有值
    
    function __construct ($name){
        $this->name= $name;
    }
    
    //添加where 條件
    function addTest($operator,$value){
        $this->comps[] = array('name'=>$this->name,'operator'=>$operator,'value'=>$value);
    }
    
    //獲取存放條件的數組
    function getComps(){
        return $this->comps;
    }
    
    function isIncomplete(){
        return empty($this->comps);
    }
}

//標識對象,主要功能就是拼接where條件語句
class IdentityObject {
    protected $currentfield = null;        //當前操作的欄位對象
    protected $fields = array();        //欄位對象集合
    private $and = null;
    private $enforce = array();            //限定的合法欄位        
    
    function __construct($field = null, array $enforce = null){
        if(!is_null($enforce)){
            $this->enforce = $enforce;
        }
        if(!is_null($field)){
            $this->field($field);
        }
    }
    
    //獲取限定的合法欄位
    function getObjectFields(){
        return $this->enforce;
    }
    
    //主要功能為設置當前需要操作的欄位對象
    function field($fieldname){
        if(!$this->isVoid()&& $this->currentfield->isIncomplete()){
            throw new \Exception("Incomplete field");
        }
        $this->enforceField($fieldname);
        if(isset($this->fields[$fieldname]){
            $this->currentfield = $this->fields[$fieldname];
        } else {
            $this->currentfield = new Field($fieldname);
            $this->fields[$fieldname] = $this->currentfield;
        }
        return $this;                    //採用連貫語法
    }
    
    //欄位集合是否為空
    function isVoid(){
        return empty($this->fields);
    }
    
    //檢查欄位是否合法
    function enforceField ($fieldname){
        if(!in_array($fieldname,$this->enforce) && !empty($this->enforce)){
            $forcelist = implode(',',$this->enforce);
            throw new \Exception("{$fieldname} not a legal field {$forcelist}");
        }
    }
    
    
    //向欄位對象添加where條件
    function eq($value){
        return $this->operator("=",$value);
    }
    
    function lt($value){
        return $this->operator("<",$value);
    }
    
    function gt($value){
        return $this->operator(">",$value);
    }
    
    //向欄位對象添加where條件
    private function operator($symbol,$value){
        if($this->isVoid){
            throw new \Exception("no object field defined");
        }
        $this->currentfield->addTest($symbol,$value);
        return $this;                                     //採用連貫語法
    }
    
    //獲取此類中所有欄位對象集合的where條件數組
    function getComps(){
        $ret = array();
        foreach($this->fields as $key => $field){
            $ret = array_merge($ret,$field->getComps());
        }
        return $ret;
    }
}


//數據映射器

class DomainObjectAssembler {
    protected static $PDO;
    
    //PersistenceFactory本例中並未實現,按原書的說法讀者自己完成
    //其主要功能就是生成相應類型的選擇工廠類、更新工廠類或Collection對象
    //在初始化的時候根據傳入的PersistenceFactory對象決定了這個類是映射那個資料庫表和領域模型的
    function __construct (PersistenceFactory $factory){
        $this->factory = $factory;
        if(!isset(self::$PDO)){
            $dsn = \woo\base\ApplicationRegistry::getDSN();
            if(is_null($dsn)){
                throw new \woo\base\AppException("NO DSN");
            }
            self::$PDO = new \PDO ($dsn);
            self::$PDO->setAttribute(\PDO::ATTR_ERRMODE,\PDO::ERRMODE_EXCEPTION);
        }
    }
    
    //獲取預處理對象,用sql語句本身做為對象數組的鍵
    function getStatement($str){
        if(!isset($this->statements[$str])){
            $this->statements[$str] = self::$PDO->prepare($str);
        }
        return $this->statements[$str];
    }
    
    //根據where條件返回一條數據
    function findOne(IdentityObject $idobj){
        $collection = $this->find($idobj);
        return $collection->next();
    }
    
    //根據where條件返回一個數據集合
    function find(IdentityObject $idobj){
        $selfact = $this->factory->getSelectionFactory();             //獲取選擇工廠對象
        list($selection,$values) = $selfact->newSelection($idobj);    //獲取sql語句
        $stmt = $this->getStatement($selection);                    //獲取預處理對象
        $stmt->execute($values);                    
        $raw = $stmt->fetchAll();
        return $this->factory->getCollection($raw);                    //還記得Collection對象嗎,下麵粘出代碼回顧一下
    }
    
    //根據where條件插入或更新一條數據
    function insert(\woo\domain\DomainObject $obj){
        $upfact = $this->factory->getUpdateFactory();        //獲取更新工廠對象
        list($update,$values) = $upfact->newUpdate($obj);    //獲取sql語句
        $stmt = $this->getStatement($update);                //獲取預處理對象
        $stmt->execute($values);
        if($obj->getId()<0){                                    
            $obj->setId(self::$PDO->lastInsertId);
        }
        $obj->markClean();
    }
    
    
}

/*
這裡在回顧一下Collection類,當然這個類和上面使用的Collection類是有差別的,因為至少這裡的Mapper類與之前相比發生了一些變化
Iterator介面定義的方法:
rewind()            指向列表開頭    
current()            返回當前指針處的元素
key()                返回當前的鍵(比如,指針的指)
next()                
valid()

下麵這個類是處理多行記錄的,傳遞資料庫中取出的原始數據和映射器進去,然後通過數據映射器在獲取數據時將其創建成對象

*/
abstract class Collection implements \Iterator{
    protected $mapper;            //數據映射器
    protected $total = 0;        //集合元素總數量
    protected $raw = array();    //原始數據
    
    private $result;
    private $pointer = 0;        //指針
    private $objects = array();    //對象集合
    
    function __construct (array $raw = null,Mapper $mapper= null){
        if(!is_null($raw)&& !is_null($mapper)){
            $this->raw = $raw;
            $this->total = count($raw);
        }
        $this->mapper = $mapper;
    }
    
    function add(\woo\domain\DmainObject $object){    //這裡是直接添加對象
        $class = $this->targetClass();
        if(!($object instanceof $class)){
            throw new Exception("This is a {$class} collection");
        }
        $this->notifyAccess();
        $this->objects[$this->total] = $object;
        $this->total ++;
    }
    
    abstract function targetClass();    //子類中實現用來在插入對象時檢查類型的
    
    protected function notifyAccess(){    //不知道幹嘛的
        
    }
    
    private function getRow($num){        //獲取集合中的單條數據,就是這裡通過數據映射器將數據創建成對象
        $this->notifyAccess();
        if($num >= $this->total || $num < 0){
            return null;
        }
        if(isset($this->objects[$num]){
            return $this->objects[$num];
        }
        if(isset($this->raw[$num]){
            $this->objects[$num] = $this->mapper->createObject($this->raw[$num]);
            return $this->objects[$num];
        }
    }
    
    public function rewind(){            //重置指針
        $this->pointer = 0;
    }
    
    public function current(){            //獲取當前指針對象
        return $this->getRow($this->pointer);
    }
    
    public function key(){                //獲取當前指針
        return $this->pointer;
    }
    
    public function next(){            //獲取當前指針對象,並將指針下移    
        $row = $this->getRow($this->pointer);
        if($row){$this->pointer ++}
        return $row;
    }
    
    public function valid(){        //驗證
        return (!is_null($this->current()));
    }
    
}

//子類
class VenueColletion extends Collection implements \woo\domain\VenueCollection{
    function targetClass(){
        return "\woo\domain\Venue";
    }
}



//客戶端
//初始化一個能夠獲取venue類型的選擇工廠類、更新工廠類或Collection對象的超級工廠類
$factory = \woo\mapper\PersistenceFactory::getFactory("woo\\domain\\Venue");    
$finder = new \woo\mapper\DomainObjectAssembler($factory);        //數據映射器
$idobj = $factory->getIdentityObject()->field('name')->eq('The Eyeball Inn');    //設置where條件語句
$collection = $finder->find($idobj);                    //根據where條件查找一個數據集合(一個Collection對象)
foreach($collection as $venue){                            
    print $venue->getName()."\n";
}

 


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 我們習慣在一個單獨的線程中額外執行一些耗時,或者執行一些可能妨礙其他任務的任務的操作,通過多個線程的合理運用,讓客戶端得到更為快速的響應方式。通過利用單獨的線程去執行 CPU 密集型和 IO 密集型的工作,讓用戶界面一直處於活躍狀態。當然,任何事物都具有兩面性,處理不當的話會誘發線程安全的問題。 創 ...
  • ResourcePath屬性 ResourcePath屬性 一、屬性介紹 獲取或設置圖像存儲路徑,預設設置為“image”,表示的ResourcePath是在程式運行路徑下的Image文件夾(bin\Debug\Image); 該屬性可以設置為Resources參數,也可以是實際路徑。使用Resou ...
  • 【增加一條新的數據】 因為使用資料庫先行的模式,所以將數據保存到資料庫的操作變得非常簡單,你只需要寫簡單的幾行代碼就能將對象的實例保存到資料庫中 你也可以使用下麵的方式,將數據保存到資料庫中 當然保存數據也是支持非同步的 【批量數據插入】 Entity Framework提供了AddRange方法,可 ...
  • 在mvvm模式的wpf項目中有個需求需要去載入解決方案的程式集,並且根據程式集去動態載入當前程式集的類,做成下拉框形式。 效果: wpf窗體: ...
  • 本期來討論下,jupyter notebook中怎樣同時安裝python2.7 和python3.x。 ...
  • 介面是什麼,抽象類又是什麼,有了抽象類,我們為什麼還要有介面呢? 抽象類 抽象類的概念: 抽象類是相對於普通類而言的,普通類是一個完善的功能類,可以直接產生實例化對象,並且在普通類中可以包含有構造方法、普通方法、static方法、常量和變數等內容。而抽象類是指在普通類的結構裡面增加抽象方法的組成部分 ...
  • 一、說明 與@Component註解功能相同,但意義不同的註解還有三個: 1)@Repository:註解在Dao實現類上 2)@Service:註解在Service實現類上 3)@Controller:註解在SpringMVC的處理器上 Bean作用域: @Scope("prototype"):用 ...
  • Python中的可變對象和不可變對象 什麼是可變/不可變對象 不可變對象,該對象所指向的記憶體中的值不能被改變。 當改變某個變數時候,由於其所指的值不能被改變,相當於把原來的值複製一份後再改變,這會開闢一個新的地址,變數再指向這個新的地址。 可變對象,該對象所指向的記憶體中的值可以被改變。 變數(準確的 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...