經常在SQL中會遇到比較復雜的SQL,很多老手也不知道怎么入手。在Postgres數據庫中有一種強大的with用法,可以分拆復雜的SQL并重新組裝,也可以當做臨時表來使用,之前也曾用with語法來實現SQL層面的遞歸。以下介紹開發中遇到的另一個例子。?
環境: Postgres 9.1.2?
評論表T1(user_id reference T2(user_id))?
用戶表T2(user_id)?
**場景:**?
需要對T1表中數據按評論數分組排序,選擇前5條記錄與表T2進行關聯,返回滿足條件的T2用戶數據,但要根據T1的排序結果來展示,也就是要顯示評論最多的5個用戶的詳細信息,并按評論數把用戶從高到低排列。?
開發的SQL:?
1.查看表T1,并排序
~~~
select user_id,count(1) from t1 group by user_id order by count(1) desc limit 5;
~~~
??[](http://static.oschina.net/uploads/space/2013/0120/154516_bhDU_267373.jpg)?
2.與表T2關聯
~~~
select * from t2 where user_id in(
select user_id from t1 group by user_id order by count(1) desc limit 5);
~~~
但是展示的T2結果雖然取出來了,但并沒有排序,沒有達到效果?
[](http://static.oschina.net/uploads/space/2013/0120/154627_SSOx_267373.jpg)?
**解決辦法:**?
1.使用兩個表的join來實現
~~~
select t2.user_id,t2.user_name,count(1) as num from t1,t2
where t1.user_id = t2.user_id
group by t2.user_id,t2.user_name
order by num desc limit 5;
~~~
2.用with來構造:
~~~
with tmp as(
select user_id,count(1) as num from t1 group by user_id order by num desc limit 5
)
select t2.*,tmp.num from t2 inner join tmp
on t2.user_id = tmp.user_id order by tmp.num desc
~~~
**分析:**?
第一種辦法需要兩張表關聯再分組取前5條,執行計劃如下:
[](http://static.oschina.net/uploads/space/2013/0120/162327_C84g_267373.jpg)
如果數據量比較大,尤其是T2表很大的時候,會消耗比較多的資源,另外如果想取用戶表里有其他的字段,也需要進行分組
第二種辦法是采用with來構造臨時表,然后再去與T2表關聯取數,執行計劃如下:?
[](http://static.oschina.net/uploads/space/2013/0120/162339_fVTc_267373.jpg)
可以看到消耗的資源相對少一點,測試中發現T2、T1表很大的時候,這種差距尤其明顯。
- 數據表
- 模式Schema
- 表的繼承和分區
- 常用數據類型
- 函數和操作符-一
- 函數和操作符-二
- 函數和操作符-三
- 索引
- 事物隔離
- 性能提升技巧
- 服務器配置
- 角色和權限
- 數據庫管理
- 數據庫維護
- 系統表
- 系統視圖
- SQL語言函數
- PL-pgSQL過程語言
- PostgreSQL 序列(SEQUENCE)
- PostgreSQL的時間-日期函數使用
- PostgreSQL 查看數據庫,索引,表,表空間大小
- 用以查詢某表的詳細 包含表字段的注釋信息
- PostgreSQL 系統表查看系統信息
- postgre存儲過程簡單實用方法
- PostgreSQL實用日常維護SQL
- PostgreSQL的時間函數使用整理
- 命令
- pg_ctl控制服務器
- initdb 初始化數據庫簇
- createdb創建數據庫
- dropdb 刪除數據庫
- createuser創建用戶
- dropuser 刪除用戶
- psql交互式工具
- psql命令手冊
- pg_dump 數據庫轉儲
- pg_restore恢復數據庫
- vacuumdb 清理優化數據庫
- reindexdb 數據庫重創索引
- createlang 安裝過程語言
- droplang 刪除過程語言
- pg_upgrade 升級數據庫簇
- 調試存儲過程
- 客戶端命令-一
- 客戶端命令-二
- 使用技巧
- PostgreSQL刪除重復數據
- postgresql 小技巧
- PostgreSQL的10進制與16進制互轉
- PostgreSQL的漢字轉拼音
- Postgres重復數據的更新一例
- PostgreSQL使用with一例
- PostgreSQL在函數內返回returning
- PostgreSQL中的group_concat使用
- PostgreSQL數據庫切割和組合字段函數
- postgresql重復數據的刪除
- PostgreSQL的遞歸查詢(with recursive)
- PostgreSQL函數如何返回數據集
- PostgreSQL分區表(Table Partitioning)應用 - David_Tang - 博客園
- PostgreSQL: function 返回結果集多列和單列的例子
- 利用pgAgent創建定時任務
- 淺談 PostgreSQL 類型轉換類似Oracle
- postgresql在windows(包括win7)下的安裝配置
- PostgreSQL簡介、安裝、用戶管理、啟動關閉、創建刪除數據庫 (2010-11-08 12-52-51)轉載▼標簽: 雜談分類: PostgreSQL
- PostgreSQL的generate_series函數應用
- PostgreSQL 8.3.1 全文檢索(Full Text Search)
- postgresql record 使用
- 備份恢復
- PostgreSQL基于時間點恢復(PITR)
- Postgresql基于時間點恢復PITR案例(二)
- Postgres邏輯備份腳本
- Postgres invalid command \N數據恢復處理