SQL Server 2008 paging methods?

前端 未结 7 554
傲寒
傲寒 2020-11-29 06:23

I have to work with a potentially large list of records and I\'ve been Googling for ways to avoid selecting the whole list, instead I want to let users select a page (like f

相关标签:
7条回答
  • 2020-11-29 06:41

    Try something like this:

    declare @page int = 2
    declare @size int = 10
    
    declare @lower int =  (@page - 1) * @size
    declare @upper int =  (@page    ) * @size
    
    select * from (
    select 
        ROW_NUMBER() over (order by some_column) lfd,
    * from your_table
    ) as t
     where lfd between @lower and @upper
     order by some_column
    
    0 讨论(0)
  • 2020-11-29 06:41

    Why not to use recommended solution:

    SELECT VALUE product FROM AdventureWorksEntities.Products AS product order by product.ListPrice SKIP @skip LIMIT @limit

    0 讨论(0)
  • 2020-11-29 06:47

    Using OFFSET

    Others have explained how the ROW_NUMBER() OVER() ranking function can be used to perform pages. It's worth mentioning that SQL Server 2012 finally included support for the SQL standard OFFSET .. FETCH clause:

    SELECT first_name, last_name, score
    FROM players
    ORDER BY score DESC
    OFFSET 40 ROWS FETCH NEXT 10 ROWS ONLY
    

    If you're using SQL Server 2012 and backwards-compatibility is not an issue, you should probably prefer this clause as it will be executed more optimally by SQL Server in corner cases.

    Using the SEEK Method

    There is an entirely different, much faster way to perform paging in SQL. This is often called the "seek method" as described in this blog post here.

    SELECT TOP 10 first_name, last_name, score
    FROM players
    WHERE (score < @previousScore)
       OR (score = @previousScore AND player_id < @previousPlayerId)
    ORDER BY score DESC, player_id DESC
    

    The @previousScore and @previousPlayerId values are the respective values of the last record from the previous page. This allows you to fetch the "next" page. If the ORDER BY direction is ASC, simply use > instead.

    With the above method, you cannot immediately jump to page 4 without having first fetched the previous 40 records. But often, you do not want to jump that far anyway. Instead, you get a much faster query that might be able to fetch data in constant time, depending on your indexing. Plus, your pages remain "stable", no matter if the underlying data changes (e.g. on page 1, while you're on page 4).

    This is the best way to implement paging when lazy loading more data in web applications, for instance.

    Note, the "seek method" is also called keyset paging.

    0 讨论(0)
  • 2020-11-29 06:48

    The following T-SQL stored procedure is a very efficient implementation of paging. THE SQL optimiser can find the first ID very fast. Combine this with the use of ROWCOUNT, and you have an approach that is both CPU-efficient and read-efficient. For a table with a large number of rows, it certainly beats any approach that I've seen using a temporary table or table variable.

    NB: I'm using a sequential identity column in this example, but the code works on any column suitable for page sorting. Also, sequence breaks in the column being used don't affect the result as the code selects a number of rows rather than a column value.

    EDIT: If you're sorting on a column with potentially non-unique values (eg LastName), then add a second column to the Order By clause to make the sort values unique again.

    CREATE  PROCEDURE dbo.PagingTest
    (
        @PageNumber int,
        @PageSize int
    )
    AS
    
    DECLARE @FirstId int, @FirstRow int
    
    SET @FirstRow = ( (@PageNumber - 1) * @PageSize ) + 1
    SET ROWCOUNT @FirstRow
    
    -- Add check here to ensure that @FirstRow is not
    -- greater than the number of rows in the table.
    
    SELECT   @FirstId = [Id]
    FROM     dbo.TestTable
    ORDER BY [Id]
    
    SET ROWCOUNT @PageSize
    
    SELECT   *
    FROM     dbo.TestTable
    WHERE    [Id] >= @FirstId
    ORDER BY [Id]
    
    SET ROWCOUNT 0
    GO 
    
    0 讨论(0)
  • 2020-11-29 06:51

    Here's an updated version of @RoadWarrior's code, using TOP. Performance is identical, and extremely fast. Make sure you have an index on TestTable.ID

    CREATE PROC dbo.PagingTest
        @SkipRows int,
        @GetRows int
    AS
    DECLARE @FirstId int
    
    SELECT   TOP (@SkipRows) 
             @FirstId = [Id]
    FROM     dbo.TestTable
    ORDER BY [Id]
    
    SELECT   TOP (@GetRows) *
    FROM     dbo.TestTable
    WHERE    [Id] >= @FirstId
    ORDER BY [Id]
    
    GO 
    
    0 讨论(0)
  • 2020-11-29 06:53

    If you use a CTE with two row_number() columns - one sorted asc, one desc, you get row numbers for paging as well as the total records by adding the two row_number columns.

    create procedure get_pages(@page_number int, @page_length int)
    as
        set nocount on;
    
        with cte as
        (
            select 
                Row_Number() over (order by sort_column desc) as row_num
                ,Row_Number() over (order by sort_column) as inverse_row_num
                ,id as cte_id
            From my_table
        )
        Select 
            row_num+inverse_row_num as total_rows
            ,*  
        from CTE inner join my_table
            on cte_id=df_messages.id
        where row_num between 
            (@page_number)*@page_length 
            and (@page_number+1)*@page_length
        order by rownumber
    
    0 讨论(0)
提交回复
热议问题