適配器模式(有時候也稱包裝樣式或者包裝)將一個類的接口適配成用戶所期待的。一個適配容許一般由於接口不兼容而不能在一塊兒工做的類工做在一塊兒。php
Target適配目標 : 該角色定義把其餘類轉換爲什麼種接口,也就是咱們的指望接口。html
Adaptee被適配者 :就是須要被適配的接口。mysql
Adapter適配器:其餘的兩個角色都是已經存在的角色,而適配器角色是須要新創建的,它用來對Adaptee與Target接口進行適配。redis
如程序數據庫有關聯mysql、mysqli、pdo、sqlite、postgresql等操做,而你須要根據狀況換數據庫操做時,可使用適配器模式統一接口,這樣代碼中除了數據庫配置以外,就不須要作而外的更改。sql
同理cache(緩存)的場景也是,不管使用memcache仍是redis等,在更換的時候都會很方便,節約時間。數據庫
注:在一些流行框架中均可以看到此模式,詳情請自行參閱框架源碼。api
代碼:緩存
<?php header('Content-Type:text/html;charset=utf-8'); /** * 適配器模式演示代碼 * Target適配目標: IDataBase接口 * Adaptee被適配者: mysql和mysql_i、postgresql的數據庫操做函數 * Adapter適配器 :mysql類和mysql_i、postgresql類 */ /** * Interface IDatabase 適配目標,規定的接口將被適配對象實現 * 約定好統一的api行爲 */ interface IDatabase { // 定義數據庫鏈接方法 public function connect($host, $username, $password, $database); // 定義數據庫查詢方法 public function query($sql); // 關閉數據庫 public function close(); } /** * Class Mysql 適配器 */ class Mysql implements IDatabase { protected $connect; // 鏈接資源 /** * 實現鏈接方法 * * @param $host host * @param $username 用戶名 * @param $password 密碼 * @param $database 數據庫名 */ public function connect($host, $username, $password, $database) { $connect = mysql_connect($host, $username, $password); mysql_select_db($database, $connect); $this->connect = $connect; //其餘操做 } /** * 實現查詢方法 * * @param $sql 須要被查詢的sql語句 * @return mi */ public function query($sql) { return mysql_query($sql); } // 實現關閉方法 public function close() { mysql_close(); } } /** * Class Mysql 適配器 */ class Mysql_i implements IDatabase { protected $connect; // 鏈接資源 /** * 實現鏈接方法 * * @param $host host * @param $username 用戶名 * @param $password 密碼 * @param $database 數據庫名 */ public function connect($host, $username, $password, $database) { $connect = mysqli_connect($host, $username, $password, $database); $this->connect = $connect; //其餘操做 } /** * 實現查詢方法 * * @param $sql 須要被查詢的sql語句 */ public function query($sql) { return mysqli_query($this->connect, $sql); } // 實現關閉 public function close() { mysqli_close($this->connect); } } /** * Class Postgresql 適配器 */ class Postgresql implements IDatabase { protected $connect; // 鏈接資源 /** * 實現鏈接方法 * * @param $host * @param $username * @param $password * @param $database */ public function connect($host, $username, $password, $database) { $this->connect = pg_connect("host=$host dbname=$database user=$username password=$password"); //其餘操做 } /** * 實現查詢方法 * * @param $sql 須要被查詢的sql語句 */ public function query($sql) { // 其餘操做 } // 實現關閉方法 public function close() { } } /** * 客戶端使用演示 * 這裏以mysql爲例 * 只要模式設計好,不論有多少種數據庫,實現和調用方式都是同樣的 * 由於都是實現的同一個接口,因此都是能夠隨意切換的 */ $host = 'localhost'; $username = 'root'; $password = 'root'; $database = 'mysql'; //$client = new Postgresql(); //$client = new Mysql(); $client = new Mysql_i(); $client->connect($host, $username, $password, $database); $result = $client->query("select * from db"); while ($rows = mysqli_fetch_array($result)) { var_dump($rows); } $client->close();
運行結果:框架
array(44) { [0]=> string(1) "%" ["Host"]=> string(1) "%" [1]=> string(4) "test" ["Db"]=> string(4) "test" [2]=> string(0) "" ["User"]=> string(0) "" [3]=> string(1) "Y" ["Select_priv"]=> string(1) "Y" [4]=> string(1) "Y" ["Insert_priv"]=> string(1) "Y" [5]=> string(1) "Y" ["Update_priv"]=> string(1) "Y" [6]=> string(1) "Y" ["Delete_priv"]=> string(1) "Y" [7]=> string(1) "Y" ["Create_priv"]=> string(1) "Y" [8]=> string(1) "Y" ["Drop_priv"]=> string(1) "Y" [9]=> string(1) "N" ["Grant_priv"]=> string(1) "N" [10]=> string(1) "Y" ["References_priv"]=> string(1) "Y" [11]=> string(1) "Y" ["Index_priv"]=> string(1) "Y" [12]=> string(1) "Y" ["Alter_priv"]=> string(1) "Y" [13]=> string(1) "Y" ["Create_tmp_table_priv"]=> string(1) "Y" [14]=> string(1) "Y" ["Lock_tables_priv"]=> string(1) "Y" [15]=> string(1) "Y" ["Create_view_priv"]=> string(1) "Y" [16]=> string(1) "Y" ["Show_view_priv"]=> string(1) "Y" [17]=> string(1) "Y" ["Create_routine_priv"]=> string(1) "Y" [18]=> string(1) "N" ["Alter_routine_priv"]=> string(1) "N" [19]=> string(1) "N" ["Execute_priv"]=> string(1) "N" [20]=> string(1) "Y" ["Event_priv"]=> string(1) "Y" [21]=> string(1) "Y" ["Trigger_priv"]=> string(1) "Y" }
從以上結果可看出,數據庫鏈接以及查詢語句都已經執行成功。函數