SQL Server query with pagination and count

前端 未结 5 1002
一生所求
一生所求 2020-12-12 13:54

I want to make a database query with pagination. So, I used a common-table expression and a ranked function to achieve this. Look at the example below.

decla         


        
5条回答
  •  失恋的感觉
    2020-12-12 14:19

    Assuming you are using MSSQL 2012, you can use Offset and Fetch which cleans up server-side paging greatly. We've found performance is fine, and in most cases better. As far as getting the total column count, just use the window function below inline...it will not include the limits imposed by 'offset' and 'fetch'.

    For Row_Number, you can use window functions the way you did, but I would recommend that you calculate that client side as (pagenumber*pagesize + resultsetRowNumber), so if you're on the 5th page of 10 results and on the third row you would output row 53.

    When applied to an Orders table with about 2 million orders, I found the following:

    FAST VERSION

    This ran in under a second. The nice thing about it is that you can do your filtering in the common table expression once and it applies both to the paging process and the count. When you have many predicates in the where clause, this keeps things simple.

    declare @skipRows int = 25,
            @takeRows int = 100,
            @count int = 0
    
    ;WITH Orders_cte AS (
        SELECT OrderID
        FROM dbo.Orders
    )
    
    SELECT 
        OrderID,
        tCountOrders.CountOrders AS TotalRows
    FROM Orders_cte
        CROSS JOIN (SELECT Count(*) AS CountOrders FROM Orders_cte) AS tCountOrders
    ORDER BY OrderID
    OFFSET @skipRows ROWS
    FETCH NEXT @takeRows ROWS ONLY;
    

    SLOW VERSION

    This took about 10 sec, and it was the Count(*) that caused the slowness. I'm surprised this is so slow, but I suspect it's simply calculating the total for each row. It's very clean though.

    declare @skipRows int = 25,
    @takeRows int = 100,
    @count int = 0
    
    
    SELECT 
        OrderID,
        Count(*) Over() AS TotalRows
    FROM Location.Orders
    ORDER BY OrderID
    OFFSET @skipRows ROWS
    FETCH NEXT @takeRows ROWS ONLY;
    

    CONCLUSION

    We've gone through this performance tuning process before and actually found that it depended on the query, predicates used, and indexes involved. For instance, the second we introduced a view it chugged, so we actually query off the base table and then join up the view (which includes the base table) and it actually performs very well.

    I would suggest having a couple of straight-forward strategies and applying them to high-value queries that are chugging.

提交回复
热议问题