設計模式之工廠模式

參考:http://www.w3cschool.cc/design-pattern/factory-pattern.html

簡介:
1.創建型模式,它提供了一種創建對象的最佳方式。
2.工廠模式中,我們在創建對象時不會對客戶端暴露創建邏輯,並且使用一個共同的接口來指向新創建的對象。
意圖:定義一個創建對象的接口,讓其子類自己決定實例化哪一個工廠類,工廠模式使其創建過程延遲到子類進行。
主要解決:主要解決接口選擇的問題。
何時使用:我們明確地計劃不同條件下創建不同實例時。
如何解決:讓其子類實現工廠接口,返回的也是一個抽象的產品。
關鍵代碼:創建過程在其子類執行。
應用實例: 1、您需要一輛汽車,可以直接從工廠裏面提貨,而不用去管這輛汽車是怎麼做出來的,以及這個汽車裏面的具體實現。 2、Hibernate 換數據庫只需換方言和驅動就可以。
優點: 1、一個調用者想創建一個對象,只要知道其名稱就可以了。 2、擴展性高,如果想增加一個產品,只要擴展一個工廠類就可以。 3、屏蔽產品的具體實現,調用者只關心產品的接口。
缺點:每次增加一個產品時,都需要增加一個具體類和對象實現工廠,使得系統中類的個數成倍增加,在一定程度上增加了系統的複雜度,同時也增加了系統具體類的依賴。這並不是什麼好事。
使用場景: 1、日誌記錄器:記錄可能記錄到本地硬盤、系統事件、遠程服務器等,用戶可以選擇記錄日誌到什麼地方。 2、數據庫訪問,當用戶不知道最後系統採用哪一類數據庫,以及數據庫可能有變化時。 3、設計一個連接服務器的框架,需要三個協議,"POP3"、"IMAP"、"HTTP",可以把這三個作爲產品類,共同實現一個接口。
注意事項:作爲一種創建類模式,在任何需要生成複雜對象的地方,都可以使用工廠方法模式。有一點需要注意的地方就是複雜對象適合使用工廠模式,而簡單對象,特別是只需要通過 new 就可以完成創建的對象,無需使用工廠模式。如果使用工廠模式,就需要引入一個工廠類,會增加系統的複雜度。

實現:

1.W3C中將實體類中的方法抽象出來,當調用時使用統一的方法。

2.使用工廠類,根據不同情況返回不同值。可使用if,也可使用switch。


最近做了一個醫院項目,數據庫封裝用到了工廠模式,下面是PHP代碼。後續會做一些java的例子。

工廠類:

class sqlOperFactory
{
    public static function getSqlOper($tableName)
    {
        switch ($tableName) {
            case "clinic":
                include 'SqlOper/ClinicOper.php';
                return new CommentOper(); // 獲得科室操作類
            case "comment":
                include 'SqlOper/CommentOper.php';
                return new CommentOper(); // 獲得評論操作類
            case "diag":
                include 'SqlOper/DiagOper.php';
                return new DiagOper(); // 獲得診斷操作類
            case "doctor":
                include 'SqlOper/DoctorOper.php';
                return new DoctorOper(); // 獲得醫生操作類
            case "favor":
                include 'SqlOper/FavorOper.php';
                return new FavorOper(); // 獲得收藏操作類
            case "orders":
                include 'SqlOper/OrdersOper.php';
                return new OrdersOper(); // 獲得預約操作類
            case "user":
                include 'SqlOper/UserOper.php';
                return new UserOper(); // 獲得用戶操作類
            case "volun":
                include 'SqlOper/VolunOper.php';
                return new VolunOper(); // 獲得志願者操作類
            default:
                return null;
        }
    }
}

數據庫操作類:這裏只貼出一個,其他的和這個類似

class ClinicOper
{
    public function create($Cname, $Cclass, $link)
    {
        $sql = "INSERT INTO `clinic` (Cname,Cclass) VALUES (?, ?)"; // 定義sql語句
        $stmt = mysqli_prepare($link, $sql); // 準備一個sql statement
        $stmt->bind_param('ss', $Cname, $Cclass); // 綁定?的參數
        $stmt->execute(); // 執行
        $errorMsg = $stmt->error; // 獲取錯誤信息
        $stmt->close(); // 關閉statement
        if ($errorMsg != "") {
            error_log($errorMsg, 0); // 如果出錯,寫入系統日誌並返回-1
            return - 1;
        }
        return 1; // 如果沒錯返回1
    }
    public function delete($Cid, $link)
    {
        $sql = "DELETE FROM `clinic` WHERE Cid=?";
        $stmt = mysqli_prepare($link, $sql);
        $stmt->bind_param('i', $Cid);
        $stmt->execute();
        $errorMsg = $stmt->error;
        $stmt->close();
        if ($errorMsg != "") {
            error_log($errorMsg, 0); // 如果出錯,寫入系統日誌並返回-1
            return - 1;
        }
        return 1; // 如果沒錯返回1
    }

    public function update()
    {}

    public function retrieveAll($link)
    {
        $i = 0;
        $sql = "SELECT Cname,Cclass FROM `clinic`";
        $stmt = mysqli_prepare($this->conn, $sql);
        $stmt->execute();
        $stmt->bind_result($Cname, $Cclass); // 將結果保存到$Cname,$Cclass
        while ($stmt->fetch()) { // 將結果保存到數組
            $array[$i] = array(
                $Cname,
                $Cclass
            );
            $i ++;
        }
        $stmt->close();
        if (!isset($array)) {
            error_log("\n----查詢全部科室失敗----\n", 0);
            return - 1;
        }
        return $array;
    }
}


實現的過程沒有將方法抽象出來,因爲水平有限,通過後期學習還要改進代碼。


發佈了45 篇原創文章 · 獲贊 6 · 訪問量 10萬+
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章