真正通用的SQL分页存储过程

关于SQL分页的问题,网上找到的一些SQL其实不能真正做到通用,他们主要是以自增长ID做为前提的。但在实际使用中,很多表不是自增长的,而且主键也不止一个字段,其实我们稍做改进就可以达到通用。这里还增加了索引,以方便大数据量的使用(代码中是先抓到临时表的,从性能上说如果采用自增长式的ID表,当然可以不使用临时表以达到性能的提升)。以下代码的SP是经过我们项目使用通过的。因朋友问及此问题,我就放在这里供大家分享,如各位大侠有更好的方法,也请告诉我。在这里先感谢各位了。

ALTER PROCEDURE [dbo].[SP_PageResult2]     
-- 获得某一页的数据 --     
@currPage int = 1, --当前页页码 (即Top currPage)     
@showColumn varchar(2000) = '*', --需要得到的字段 (即 column1,column2,......)     
@tabName varchar(2000), --需要查看的表名 (即 from table_name)     
@strCondition varchar(2000) = '', --查询条件 (即 where condition......) 不用加where关键字     
@ascColumn varchar(100) = '', --排序的字段名 (即 order by column asc/desc)     
@bitOrderType bit = 0, --排序的类型 (0为升序,1为降序)     
--@pkColumn varchar(50) = '', --主键名称     
@pageSize int = 20 --分页大小     
--  SP_PageResult2 '1','*','Bom','1=1','Ancestor',0,30    
AS     
BEGIN -- 存储过程开始     
-- 该存储过程需要用到的几个变量 --     
DECLARE @strTemp varchar(1000)     
DECLARE @strSql varchar(4000) --该存储过程最后执行的语句     
DECLARE @strOrderType varchar(1000) --排序类型语句 (order by column asc或者order by column desc)     
DECLARE @RIDorder varchar(100)    
BEGIN     
IF @bitOrderType = 1 -- bitOrderType=1即执行降序     
 BEGIN     
 SET @strOrderType = ' ORDER BY '+@ascColumn+' DESC'     
 SET @strTemp = '>(SELECT max'        -- 因为降降得升,所以这里依然是>(SELECT max ,而不是<(SELECT min
 SET @RIDorder  = ' ORDER BY RID ASC'  -- 因为降降得升。所以这里依然是asc  
 END     
ELSE     
 BEGIN     
 SET @strOrderType = ' ORDER BY '+@ascColumn+' ASC'     
 SET @strTemp = '>(SELECT max'    
 SET @RIDorder  = ' ORDER BY RID ASC'  
 END     
  
  
IF @currPage = 1 -- 如果是第一页     
BEGIN     
IF @strCondition != ''    
 BEGIN  
  SET @strSql = 'SELECT  row_number() over (' + @strOrderType +') AS RID,'+@showColumn+' INTO #T FROM '+@tabName+ ' WHERE '+@strCondition+' '+@strOrderType +'; create index I1 on #T(RID);'  
  SET @strSql = @strSql + ' SELECT TOP '+STR(@pageSize)+@showColumn+' FROM #T ' + @RIDorder     
 
 END  
ELSE     
 BEGIN  
  SET @strSql = 'SELECT  row_number() over (' + @strOrderType +') AS RID,'+@showColumn+' INTO #T FROM '+@tabName+ ' '+@strOrderType +'; create index I1 on #T(RID);'  
  SET @strSql = @strSql + ' SELECT TOP '+STR(@pageSize)+@showColumn+' FROM #T ' + @RIDorder
 
 END  
END     
    
ELSE -- 其他页     
BEGIN    
   
IF @strCondition !=''  
 BEGIN    
  SET @strSql = 'SELECT  row_number() over (' + @strOrderType +') AS RID,'+@showColumn+' INTO #T FROM '+@tabName+ ' WHERE '+@strCondition+' '+@strOrderType +'; create index I1 on #T(RID);'     
  SET @strSql = @strSql + 'SELECT TOP '+STR(@pageSize)+' '+@showColumn+' FROM '+'#T'+     
  ' WHERE '+@strCondition+' AND '+'RID'+@strTemp+'('+'RID'+')'+' FROM (SELECT TOP '+STR((@currPage-1)*@pageSize)+     
  ' '+'RID'+' FROM '+'#T'+@RIDorder+') AS TabTemp)'+@RIDorder    
 END   
ELSE   
 BEGIN     
  SET @strSql = 'SELECT  row_number() over (' + @strOrderType +') AS RID,'+@showColumn+' INTO #T FROM '+@tabName+' ' + @strOrderType  +'; create index I1 on #T(RID);'      
  SET @strSql = @strSql + 'SELECT TOP '+STR(@pageSize)+' '+@showColumn+' FROM '+'#T'+     
  ' WHERE '+'RID'+@strTemp+'('+'RID'+')'+' FROM (SELECT TOP '+STR((@currPage-1)*@pageSize)+' '+'RID'+     
  ' FROM '+'#T'+@RIDorder+') AS TabTemp)'+@RIDorder    
 END   
END     
    
END     
--IF @strCondition != ''   
-- set  @strSql=@strSql+';select count(*) from '+ @tabName +' where '+ @strCondition +';DROP TABLE #T'    
--else  
 set  @strSql=@strSql+';select count(*) from '+ '#T' + ';DROP TABLE #T'    
print @strSql  
exec (@strSql)     
    
END -- 存储过程结束 
View Code

下图为调用结果,从结果来看。正确的。

原文地址:https://www.cnblogs.com/Geton/p/3794241.html