摘要:聚合函數(shù)在中,有一些用來統(tǒng)計匯總的函數(shù),被稱作聚合函數(shù),如等。方法其它方法如之類的編寫就不一一展示了,代碼請看聚合函數(shù)。如何獲取總數(shù)當(dāng)然是使用上面講到的聚合函數(shù)來處理。
where 相關(guān)的子句構(gòu)造完成后,我們繼續(xù)構(gòu)造其它子句。這一篇我們進行聚合函數(shù)、分組、排序等子句的構(gòu)造。
聚合函數(shù)在 SQL 中,有一些用來統(tǒng)計、匯總的函數(shù),被稱作聚合函數(shù),如 SUM、COUNT、AVG 等。
使用 select() 方法時,我們可以用 select("COUNT(id)") 這種寫法來使用聚合函數(shù),但是這種方式有缺點:
語義上并不直觀
在防止關(guān)鍵字沖突時需要手動添加引號(參見第三篇條件查詢)
拿到聚合數(shù)據(jù)需要一串繁瑣的方法調(diào)用
為了更方便的獲得聚合數(shù)據(jù),我們需要為其多帶帶編寫方法。
getList() 方法獲得某一列的方法可以由 PDO::FETCH_COLUMN 來完成。
基類添加 getList() 方法:
public function getList($field) { $this->_cols_str = " ".self::_quote($field)." "; $this->_buildQuery(); $this->_execute(); // 獲取一列數(shù)據(jù) return $this->_pdoSt->fetchAll(PDO::FETCH_COLUMN, 0); }count() 方法
基類添加 count() 方法:
public function count($field = "*") { // 判斷是否時 * // 非 * 時給字段添加引號 if(trim($field) != "*") { $field = self::_quote($field); } // 構(gòu)造列查詢字符串 $this->_cols_str = " COUNT(".$field.") AS count_num "; // 取結(jié)果 return $this->row()["count_num"]; }
構(gòu)造 SQL SELECT COUNT(id) FROM test_table;
$results = $driver->table("test_table") ->count("id");
由于聚合函數(shù)方法和 get()、row() 方法一樣是取結(jié)果的,鏈?zhǔn)秸{(diào)用時切記要放到最后執(zhí)行。
其它方法sum()、avg() 等方法沒有 COUNT("*") 這樣的場景,比 count() 方法的編寫更簡單。
sum() 方法:
public function sum($field) { $this->_cols_str = " SUM(".self::_quote($field).") AS sum_num "; return $this->row()["sum_num"]; }
其它方法如 avg()、max() 之類的編寫就不一一展示了,代碼請看 WorkerF - 聚合函數(shù) 。
分組:group by 和 havinggroup by 子句指定了要分組的字段,可以是一個或多個 (用逗號隔開)。
having 子句和 group by 子句一起使用,作為分組的篩選條件,可以為空,除了關(guān)鍵字外,語法和 where 子句基本相同。
基類新增 groupBy() 方法:
public function groupBy($field) { // 是否初次調(diào)用 ? if($this->_groupby_str == "") { $this->_groupby_str = " GROUP BY ".self::_wrapRow($field); } else { // 非初次調(diào)用(多個分組字段),使用逗號分隔 $this->_groupby_str .= " , ".self::_wrapRow($field); } return $this; }
基類新增 having()、orHaving() 方法:
public function having() { $operator = "AND"; // 初次調(diào)用 ? if($this->_having_str == "") { $this->_having_str = " HAVING "; } else { $this->_having_str .= " ".$operator." "; } // 和 where 子句一樣進行條件構(gòu)造 $this->_condition_constructor(func_num_args(), func_get_args(), $this->_having_str); return $this; } public function orHaving() { $operator = "OR"; if($this->_having_str == "") { $this->_having_str = " HAVING "; } else { $this->_having_str .= " ".$operator." "; } $this->_condition_constructor(func_num_args(), func_get_args(), $this->_having_str); return $this; }
這里我們也留一個處理原生字符串的 havingRaw() 方法 (手動填寫數(shù)據(jù),不進行數(shù)據(jù)綁定):
public function havingRaw($string) { $this->_having_str = " HAVING ".$string." "; return $this; }
構(gòu)造 SQL SELECT id, SUM(price) FROM test_table GROUP BY price HAVING SUM(price) > 1000;:
$results = $driver->table("test_table") ->select("id", "SUM(price)") ->groupBy("price") ->having("SUM(price)", ">", 1000) ->get(); // 使用 havingRaw() $results = $driver->table("test_table") ->select("id", "SUM(price)") ->groupBy("price") ->havingRaw("SUM(price) > 1000") ->get();排序
排序就很簡單了,固定的語法和正序、倒序兩個模式,多個字段排序時使用逗號隔開:
public function orderBy($field, $mode = "ASC") { $mode = strtoupper($mode); if ( ! in_array($mode, ["ASC", "DESC"])) { throw new InvalidArgumentException("Error order by mode"); } // 初次調(diào)用? if($this->_orderby_str == "") { $this->_orderby_str = " ORDER BY ".self::_wrapRow($field)." ".$mode; } else { // 多個排序字段時,逗號隔開 $this->_orderby_str .= " , ".self::_wrapRow($field)." ".$mode; } return $this; }
構(gòu)造 SQL SELECT * FROM tes_table ORDER BY price DESC, id ASC;:
$results = $driver->table("test_table") ->orderBy("price", "DESC") ->orderBy("id", "ASC") ->get();limit 和 分頁 limit 子句
標(biāo)準(zhǔn) SQL 中的 limit 子句是 limit、offset 關(guān)鍵字一起使用的,Mysql 中有 LIMIT 0, 10 的簡寫形式,但是 PostgreSql 和 Sqlite 并不適用。所以我們選用 limit、offset 語法:
public function limit($offset, $step) { $this->_limit_str = " LIMIT ".$step." OFFSET ".$offset." "; return $this; }
構(gòu)造 SQL SELECT * FROM test_table LIMIT 10 OFFSET 1;:
$results = $driver->table("test_table") ->limit(1, 10) ->get();分頁
在數(shù)據(jù)請求時會遇到請求資源數(shù)據(jù)量大的問題,對于這個問題,普遍的解決方案就是分頁。
有了 limit() 方法,可以進行分頁功能的實現(xiàn)了。
為了靈活的訪問分頁,我們要返回以下的數(shù)據(jù):
數(shù)據(jù)的總數(shù)
每頁的數(shù)據(jù)個數(shù)
當(dāng)前頁
下一頁
上一頁
第一頁
最后一頁
當(dāng)前頁的數(shù)據(jù)集合
對于獲取數(shù)據(jù)的總數(shù),這里有一些問題。
如何獲取總數(shù)?當(dāng)然是使用上面講到的聚合函數(shù) count() 來處理。但是,使用了 count() 方法后相當(dāng)于進行了一次 SQL 的構(gòu)造和執(zhí)行 (執(zhí)行后會將構(gòu)造字符串設(shè)置為初始狀態(tài)),那么還如何進行當(dāng)頁數(shù)據(jù)集合的獲取呢?
兩次執(zhí)行
當(dāng)然是有解決方案的,就是構(gòu)造兩次。
我們的查詢構(gòu)造器結(jié)構(gòu)每次新建一個實例就會得到一個 PDO 的連接,所以為了構(gòu)造兩次而新建實例的話,代價太大,那么如何在一個實例中實現(xiàn)兩次執(zhí)行?
回顧上一篇,含有子查詢的 SQL 構(gòu)造時經(jīng)過了兩次構(gòu)造,對構(gòu)造字符串進行了保護、恢復(fù)。那么換到分頁中,還是構(gòu)造兩次 SQL,對構(gòu)造字符串進行保護、恢復(fù),區(qū)別就是:因為要執(zhí)行兩次,所以要對綁定的數(shù)據(jù)也進行保護、恢復(fù)。
基類中添加綁定數(shù)據(jù)保存、恢復(fù)的方法:
// 保存綁定數(shù)據(jù) protected function _storeBindParam() { return $this->_bind_params; } // 恢復(fù)綁定數(shù)據(jù) protected function _reStoreBindParam($bind_params) { $this->_bind_params = $bind_params; }
基類中添加 paginate() 方法:
public function paginate($step, $page = NULL) { // 保存構(gòu)造字符串和綁定數(shù)據(jù) $store = $this->_storeBuildAttr(); $bind_params = $this->_storeBindParam(); // 獲取數(shù)據(jù)總數(shù) $count = $this->count(); // 恢復(fù)構(gòu)造字符串和綁定數(shù)據(jù) $this->_reStoreBuildAttr($store); $this->_reStoreBindParam($bind_params); // 創(chuàng)建分頁數(shù)據(jù) $page = $page ? $page : 1; // 第一頁 $this->limit($step * ($page - 1), $step); $rst["total"] = $count; $rst["per_page"] = $step; $rst["current_page"] = $page; $rst["next_page"] = ($page + 1) > ($count / $step) ? NULL : ($page + 1); $rst["prev_page"] = ($page - 1) < 1 ? NULL : ($page - 1); $rst["first_page"] = 1; $rst["last_page"] = $count / $step; $rst["data"] = $this->get(); // 返回結(jié)果 return $rst; }
測試一下,獲取從第五頁的數(shù)據(jù),每頁 10 條數(shù)據(jù):
$results = $driver->table("test_table") ->paginate(10, 5);
Just do it!
文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請注明本文地址:http://m.specialneedsforspecialkids.com/yun/30826.html
摘要:而在項目開發(fā)中,我們想要的是一個更好用的可維護的工具,此時,對代碼的封裝模塊化就顯得尤為重要,于是出現(xiàn)了兩種方案查詢構(gòu)造器,對象關(guān)系映射。典型環(huán)境下按照一般的查詢構(gòu)造器處理就行。 文章目錄 寫一個特殊的查詢構(gòu)造器 - (前言) 寫一個特殊的查詢構(gòu)造器 - (一、程序結(jié)構(gòu),基礎(chǔ)封裝) 寫一個特殊的查詢構(gòu)造器 - (二、第一條語句) 寫一個特殊的查詢構(gòu)造器 - (三、條件查詢) 寫一個特殊...
閱讀 3592·2023-04-26 01:43
閱讀 2979·2021-10-14 09:42
閱讀 5461·2021-09-30 09:59
閱讀 2182·2021-09-04 16:40
閱讀 1214·2019-08-30 15:52
閱讀 832·2019-08-29 17:09
閱讀 2002·2019-08-26 13:37
閱讀 3438·2019-08-26 10:20