mysql、mssql及oracle分頁查詢方法詳解_資料庫其它

來源:互聯網
上載者:User

本文執行個體講述了mysql、mssql及oracle分頁查詢方法。分享給大家供大家參考。具體分析如下:

分頁查詢在web開發中是最常見的一種技術,最近在通過查資料,有一點自己的心得

一、mysql中的分頁查詢

註:

m=(pageNum-1)*pageSize;n= pageSize;

pageNum是要查詢的頁碼,pageSize是每次查詢的資料量,

方法一:

select * from table order by id limit m, n;

該語句的意思為,查詢m+n條記錄,去掉前m條,返回後n條記錄。無疑該查詢能夠實現分頁功能,但是如果m的值越大,查詢的效能會越低(越後面的頁數,查詢效能越低),因為MySQL同樣需要掃描過m+n條記錄。

方法二:

select * from table where id > #max_id# order by id limit n;

該查詢每次會返回n條記錄,卻無需像方式1掃描過m條記錄,在大資料量的分頁情況下,效能可以明顯好於方式1,但該分頁查詢必須要每次查詢時拿到上一次查詢(上一頁)的一個最大id(或最小id)。該查詢的問題就在於,我們有時沒有辦法拿到上一次查詢(上一頁)的最大id(或最小id),比如當前在第3頁,需要查詢第5頁的資料,該查詢方法便愛莫能助了。

方法三:

為了避免能夠實現方式二不能實現的查詢,就同樣需要使用到limit m, n子句,為了效能,就需要將m的值儘力的小,比如當前在第3頁,需要查詢第5頁,每頁10條資料,當前第3頁的最大id為#max_id#:

select * from table where id > #max_id# order by id limit 20,10;

其實該查詢方式是部分解決了方式二的問題,但如果當前在第2頁,需要查詢第100頁或1000頁,效能仍然會較差。

方法四:

複製代碼 代碼如下:
select * from table as a inner join (select id from table order by id limit m, n) as b on a.id = b.id order by a.id;

該查詢同方式一 一樣,m的值可能很大,但由於內部的子查詢只掃描了欄位id,而不是整張表,所以效能要強於方式一查詢,並且該查詢能夠解決方式二和方式三不能解決的問題。

方式五:

複製代碼 代碼如下:
select * from table where id > (select id from table order by id limit m, 1) limit n;

該查詢方式同方式四,同樣通過子查詢掃描欄位id,效果同方式四。至於效能的話,方式五的效能會略好於方式四,因為方式5不需要在進行表的關聯,而是一個簡單的比較。

二、Sql Server分頁查詢

方法一:

適用於 SQL Server 2000/2005

SELECT TOP 頁大小 * FROM table1 WHERE id NOT IN      (      SELECT TOP 頁大小*(頁數-1) id FROM table1 ORDER BY id      ) ORDER BY id

方法二:

適用於 SQL Server 2000/2005

--順序寫法:

 SELECT TOP 頁大小 * FROM table1 WHERE id >= ( SELECT ISNULL(MAX(id),0)  FROM  ( SELECT TOP 頁大小*(頁數-1)+1 id FROM table1 ORDER BY id ) A ) ORDER BY id

--降序寫法:

 SELECT TOP 頁大小 * FROM table1 WHERE id <= ( SELECT ISNULL(MIN(id),0)  FROM  ( SELECT TOP 頁大小*(頁數-1)+1 id FROM table1 ORDER BY id Desc ) A ) ORDER BY id Desc

方法三:

適用於 SQL Server 2005

 SELECT TOP 頁大小 *  FROM      (     SELECT ROW_NUMBER() OVER (ORDER BY id) AS RowNumber,* FROM table1     ) A WHERE RowNumber > 頁大小*(頁數-1)

說明,頁大小:每頁的行數;頁數:第幾頁。使用時,請把“頁大小”和“頁大小*(頁數-1)”替換成數字。

其它的方案:如果沒有主鍵,可以用暫存資料表,也可以用方案三做,但是效率會低。
建議最佳化的時候,加上主鍵和索引,查詢效率會提高。

通過SQL 查詢分析器,顯示比較:我的結論是:

分頁方案二:(利用ID大於多少和SELECT TOP分頁)效率最高,需要拼接SQL語句
分頁方案一:(利用Not In和SELECT TOP分頁) 效率次之,需要拼接SQL語句
分頁方案三:(利用SQL的遊標預存程序分頁) 效率最差,但是最為通用

三、oracle分頁查詢

方法一:

SELECT * FROM ( SELECT A.*, ROWNUM RN FROM   (SELECT * FROM tab) A    WHERE ROWNUM <= 40 )      WHERE RN >= 21;

這個分頁比下面的執行時間少,效率高。當資料量較大時oracle會自動最佳化!

方法二:

select * from (select c.*,rownum rn from tab c) where rn between 21 and 40

對比這兩種寫法,絕大多數的情況下,第一個查詢的效率比第二個高得多。

這是由於CBO最佳化模式下,Oracle可以將外層的查詢條件推到內層查詢中,以提高內層查詢的執行效率。

對於第一個查詢語句,第二層的查詢條件WHERE ROWNUM <= 40就可以被Oracle推入到內層查詢中,這樣Oracle查詢的結果一旦超過了ROWNUM限制條件,就終止查詢將結果返回了。

而第二個查詢語句,由於查詢條件BETWEEN 21 AND 40是存在於查詢的第三層,而Oracle無法將第三層的查詢條件推到最內層

(即使推到最內層也沒有意義,因為最內層查詢不知道RN代表什麼)。因此,對於第二個查詢語句,Oracle最內層返回給中介層的是所有滿足條件的資料,而中介層返回給最外層的也是所有資料。資料的過濾在最外層完成,顯然這個效率要比第一個查詢低得多。

上面分析的查詢不僅僅是針對單表的簡單查詢,對於最內層查詢是複雜的多表聯集查詢或最內層查詢包含排序的情況一樣有效。

希望本文所述對大家的資料庫程式設計有所協助。

聯繫我們

該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在5個工作日內處理。

如果您發現本社區中有涉嫌抄襲的內容,歡迎發送郵件至: info-contact@alibabacloud.com 進行舉報並提供相關證據,工作人員會在 5 個工作天內聯絡您,一經查實,本站將立刻刪除涉嫌侵權內容。

A Free Trial That Lets You Build Big!

Start building with 50+ products and up to 12 months usage for Elastic Compute Service

  • Sales Support

    1 on 1 presale consultation

  • After-Sales Support

    24/7 Technical Support 6 Free Tickets per Quarter Faster Response

  • Alibaba Cloud offers highly flexible support services tailored to meet your exact needs.