注册 登录
编程论坛 ASP.NET技术论坛

存储过程自定义分页详解

hebingbing 发布于 2008-04-05 04:52, 12089 次点击
*/ --------------------------------------------------------------------------------------
*/ 出自: 编程中国 https://www.bc-cn.net
*/ 作者: hebingbing
*/ 时间: 2008-4-5 04:30 编程论坛首发
*/ 声明: 光看我这么晚了还在工作,转载这段文字应该保留吧……
*/ --------------------------------------------------------------------------------------
废话:清明节,同学回家的回家,旅游的旅游……我离家远是不可能回家了,旅游吧不感兴趣,觉得还不如看一场电影……呵呵,从小不喜欢旅游观光……
转入正题:大家都知道中的Gridview。datalist等都可以自定义分页,但是当你翻页的时候,数据表中的所有数据都会加载到内存,重新绑定,当然要是数据量小的话,这是可以的,我们也很乐意用,原因简单因为方便,但是要是数据量是999999999999……,在信息爆炸的这个时代海量数据是经常的时,那么这些控件自带的分页就显得有些……
解决这个问题办法就是自己动手……不多废话了,看代码:
1.首先我是用存储过程来解决的,要弄懂这个问题,首先要从存储过程下手,代码如下:


程序代码:
CREATE proc getdataset
@TableList Varchar(200)='*',--搜索表的字段,比如:’id,datatime,job‘,用逗号隔开
@TableName Varchar(30), --搜索的表名
@SelectWhere Varchar(500)='',--搜索条件,这里不用写where,比如:job=’teacher‘and class='2'
@SelectOrderId Varchar(20),--表主键字段名。比如:id
@SelectOrder Varchar(200)='', --排序,可以使用多字段排序但主键字段必需在最前面.也可以不写,比如:order by class asc
@intPageNo int=1, --页号
@intPageSize int=10 ,--每页显示数
@RecordCount int OUTPUT  --总记录数(存储过程输出参数)
as  
   
declare @TmpSelect      NVarchar(600)  
declare @Tmp     NVarchar(600)  

set nocount on--关闭计数

set @TmpSelect = 'select @RecordCount = count(*) from '+@TableName+' '+@SelectWhere

execute sp_executesql
@TmpSelect,    --执行上面的sql语句
N'@RecordCount int OUTPUT' ,   --执行输出数据的sql语句,output出总记录数
@RecordCount  OUTPUT

 
  if (@RecordCount = 0)    --如果没有贴子,则返回零
       return 0
      
   /*判断页数是否正确*/
  if (@intPageNo - 1) * @intPageSize > @RecordCount   --页号大于总页数,返回错误
     return (-1)
set nocount off--打开计数
if @SelectWhere != ''
begin
set @TmpSelect = 'select top '+str(@intPageSize)+' '+@TableList+' from '+@TableName+' where '+@SelectOrderId+' not in(select top '+str((@intPageNo-1)*@intPageSize)+' '+@SelectOrderId+' from '+@TableName+' '+@SelectWhere +' '+@SelectOrder+') and '+@SelectWhere +' '+@SelectOrder
end
else
begin
set @TmpSelect = 'select top '+str(@intPageSize)+' '+@TableList+' from '+@TableName+' where '+@SelectOrderId+' not in(select top '+str((@intPageNo-1)*@intPageSize)+' '+@SelectOrderId+' from '+@TableName+' '+@SelectOrder+') '+@SelectOrder
end
execute sp_executesql @TmpSelect
return(@@rowcount)
GO

其实代码也很简单,学编程的人基本上都是懂数据库的,这个存储过程估计不是问题。
其他的代码我都做了解释,有颜色的那段我没有解释,我在这里解释一下。其实也很简单,大家来看:
select top '+str((@intPageNo-1)*@intPageSize)+' '+@SelectOrderId+' from '+@TableName+' '+@SelectWhere +' '+@SelectOrder+'
这段代码的执行结果是什么了,是不是当前页前面的主键的集合啊,现在我们从所有的表中选出主键的值不在这个结果的之内的pagesize个记录不就是当前页的内容了吗?
2.aspx页面就不用再将了吧?我这里将代码写上:
程序代码:
<%@ Page Language="C#" AutoEventWireup="true" CodeFile="aa.aspx.cs" Inherits="_Default" %>

<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www. xmlns="http://www. >
<head runat="server">
    <title>无标题页</title>
</head>
<body>
    <form id="form1" runat="server">
    <div>
        <asp:GridView ID="GridView1" runat="server" AutoGenerateColumns="False" Height="180px" Width="867px">
            <Columns>
                <asp:BoundField DataField="job_id" HeaderText="job_id" />
                <asp:BoundField DataField="job_desc" HeaderText="job_desc" />
                <asp:BoundField DataField="max_lvl" HeaderText="max_lxl" />
            </Columns>
        </asp:GridView>
   
    </div>
        &nbsp; &nbsp;<asp:HyperLink ID="hylfirst" runat="server">首页</asp:HyperLink>
        &nbsp;
        <asp:HyperLink ID="hylprev" runat="server">上一页</asp:HyperLink>
        &nbsp;
        <asp:HyperLink ID="hylnext" runat="server">下一页</asp:HyperLink>
        <asp:HyperLink ID="hylend" runat="server">尾页</asp:HyperLink>
        &nbsp; &nbsp; &nbsp; &nbsp;第<asp:Label ID="lbRow" runat="server" Text="Label"></asp:Label>页,
        &nbsp; 共<asp:Label ID="lbpage" runat="server" Text="Label"></asp:Label>页,共<asp:Label
            ID="lbRecord" runat="server" Text="Label"></asp:Label>条记录,转到<asp:TextBox ID="txtlink"
                runat="server" Width="29px"></asp:TextBox>
        页<asp:LinkButton ID="link" runat="server" OnClick="link_Click" TabIndex="1">转到</asp:LinkButton>
    </form>
</body>
</html>


3.cs页面其实也每页什么好讲的,也就是一些常用的代码罢了……我把代码加上,大家看看,要是有疑问的可以回复我再解释:
程序代码:
using System;
using System.Data;
using System.Configuration;
using System.Collections;
using System.Web;
using System.Web.Security;
using System.Web.UI;
using System.Web.UI.WebControls;
using System.Web.UI.WebControls.WebParts;
using System.Web.UI.HtmlControls;
using System.Data.SqlClient;

public partial class _Default : System.Web.UI.Page
{
    protected void Page_Load(object sender, EventArgs e)
    {

        this.bind();
        
    }

    protected void link_Click(object sender, EventArgs e)
    {
        int page = Convert.ToInt32(txtlink.Text);
        Response.Redirect("aa.aspx?CurrentPage="+page+"");
    }
    public void bind()
    {
        int sumPage;
        int pageNo = 1;
        int pageSize = 3;
        if (Request.QueryString["CurrentPage"] == null)
        {
            pageNo = 1;
        }
        else
        {
            pageNo = Int32.Parse(Request.QueryString["CurrentPage"]);
        }

        SqlConnection conn = new SqlConnection(ConfigurationManager.AppSettings["ConStr"]);
        SqlDataAdapter da = new SqlDataAdapter();
        da.SelectCommand = new SqlCommand();
        da.SelectCommand.Connection = conn;
        da. = "getdataset";
        da. = CommandType.StoredProcedure;
        da.SelectCommand.Parameters.Add("@TableList", SqlDbType.VarChar, 200).Value = "job_id,job_desc,max_lvl";
        da.SelectCommand.Parameters.Add("@TableName", SqlDbType.VarChar, 30).Value = "jobs";
        //da.SelectCommand.Parameters.Add("@SelectWhere", SqlDbType.VarChar, 500).Value = "where d=1";
        da.SelectCommand.Parameters.Add("@SelectOrderId", SqlDbType.VarChar, 20).Value = "job_id";
        da.SelectCommand.Parameters.Add("@SelectOrder", SqlDbType.VarChar, 200).Value = "order by min_lvl asc";
        da.SelectCommand.Parameters.Add("@intPageNo", SqlDbType.Int).Value = pageNo;
        da.SelectCommand.Parameters.Add("@intPageSize", SqlDbType.Int).Value = pageSize;
        da.SelectCommand.Parameters.Add("@RecordCount", SqlDbType.Int).Direction = ParameterDirection.Output;
        da.SelectCommand.Parameters.Add("RowCount", SqlDbType.Int).Direction = ParameterDirection.ReturnValue;
        DataSet ds = new DataSet();
        da.Fill(ds, "jobs");
        GridView1.DataSource = ds;
        GridView1.DataBind();
        Int32 RecordCount = (Int32)da.SelectCommand.Parameters["@RecordCount"].Value; //求出总记录数,该值是output出来的值
        Int32 RowCount = (Int32)da.SelectCommand.Parameters["RowCount"].Value;         //求出当前页中的记录数,在最后一页不等于pagesize,
        lbRecord.Text = RecordCount.ToString();
        lbRow.Text = pageNo.ToString();
        sumPage = (Int32)RecordCount / pageSize;
        if (RecordCount % pageSize > 0)
        {
            sumPage = sumPage + 1;
        }
        lbpage.Text = sumPage.ToString();
        if (pageNo > 1)
        {
            hylfirst.NavigateUrl = "aa.aspx?CurrentPage=1";
            hylprev.NavigateUrl = string.Concat("aa.aspx?CurrentPage=", "", pageNo - 1);
        }
        else
        {
            hylprev.NavigateUrl = "";
            hylfirst.NavigateUrl = "";
            hylfirst.Visible = false;
            hylprev.Visible = false;
        }
        if (pageNo < sumPage)
        {
            hylend.NavigateUrl = string.Concat("aa.aspx?CurrentPage=", "", sumPage);
            hylnext.NavigateUrl = string.Concat("aa.aspx?CurrentPage=", "", pageNo + 1);
        }
        else
        {
            hylnext.NavigateUrl = "";
            hylend.NavigateUrl = "";
            hylend.Visible = false;
            hylnext.Visible = false;
        }

    }
}


就这样吧。要是大家有疑问,回帖我们再讨论,在研究……
对了,我将实现的效果图给大家传上来看看:
30 回复
#2
淡漠的茶2008-04-05 11:24
这个分页存储过程有些缺点
1.没有充分利用索引
2.没有把第一页情况独立出来,因为很多网站被访问的最多的是第一页,而不是下一页,所以第一页的情况一定要独立出来
3.SQL中是尽量避免使用in 与 not in的,效率并不是很高,这个存储过程有些情况下是不需要in not in的
4.这个存储过程有个很明显的缺点,就是排序字段必须是主键字段,而且只能有一个,不能多排,本来大部分分页存储中 in  not in是用在非主键情况下的,这里却结合主键,一般主键都会建索引的,一般都是采用 > <的

不过对于简单要求分页,这个也够了

[[it] 本帖最后由 淡漠的茶 于 2008-4-5 11:26 编辑 [/it]]
#3
hebingbing2008-04-05 13:21
首先感谢ls的兄弟指点……
但是楼上所说的这几点好像也不是很对……
1.我的存储过程是没有充分利用索引,这点ls说的很好,我也没有考虑的,因为我在写这个的时候在网上也查看了一番,借鉴了一下别人的思想,我看到别人的也都没有怎么用索引,所以我也没有考虑到要用索引,这是这个不足之处。
2.至于第一页没有分离出来的情况,我倒是觉得是ls多虑,因为当是第一页的时候(select top '+str((@intPageNo-1)*@intPageSize)+' '+@SelectOrderId+' from '+@TableName+' '+@SelectWhere +' '+@SelectOrder+')这句根本就相当于不执行,相当于我分离出了第一页,相反我倒是觉得这点却是我写的这个存储过程里精彩的地方,避免了像很多程序那样分专门写出第一页的情况。
3.你所说的这里很多地方不必要用in/not in,我估计你说的是应该用EXISTS/NOT EXISTS,但是这里要主要的是如果大表在外小表在内的话in/not in的效率是要高于EXISTS/NOT EXISTS的,当然要是小表在外,大表在内的话EXISTS/NOT EXISTS是要比in/not in效率要高,况且not in和NOT EXISTS根本就不一样,如果(select top '+str((@intPageNo-1)*@intPageSize)+' '+@SelectOrderId+' from '+@TableName+' '+@SelectWhere +' '+@SelectOrder+')查询的结果是空值NOT IN永远是0,因为NULL代表“未知”,任何值和NULL比较永远是false。
4.这也是关键点,如你所说如果用"><"来半段的话,那么主键必须是int型的,其他像字符型的。字符串型等的那是不是用起来不太……,而我的这个存储过程正是解决了这个问题,主键可以是任何类型的。不必拘束与主键必须是数值型的问题。再者我的这个存储过程并不是排序字段必须是主键,并且并也不是只能有一个,你可以任意加。而你说的那样用  > < 来解决问题时排序字段才必须是主键,变量@SelectOrder 不就是自己想要的排序吗?

[[it] 本帖最后由 hebingbing 于 2008-4-5 13:26 编辑 [/it]]
#4
风风风风2008-04-05 13:50
提示: 作者被禁止或删除 内容自动屏蔽,只有管理员可见
#5
风风风风2008-04-05 14:00
提示: 作者被禁止或删除 内容自动屏蔽,只有管理员可见
#6
淡漠的茶2008-04-05 15:01
不好意思 @SelectOrder 这个字段的意思我没仔细看
呵呵,我以为是存储排序方式呢,第4点我收回
第3点我坚持,我说的第三点并不是指EXISTS,在第四点中楼主说主键必须是int型,也可以是datetime的吗,呵呵,我坚持第三点就是将楼主说的这些特殊类型,将他们独立出来,遇到这些特殊类型 > <在索引条件下效率要比 not in高这点是不用置疑的,而且这类特殊类型也是常用来排序的,我认为将这些区分出来是必要的,对于简单分页甚至可以不要下面的not in(在只有一种特殊类型排序下)
,这点从SQL执行顺序上可以分析出来,SQL先WHERE 再select 再order ,使用not 就会置将where全局两次,而> <方法一般先会缩小一个范围,再从这个范围选取 一个大于或小于主键的范围,另外in导致一次遍历,如果页码过百,这个需要遍历对比的序列将会很长
至于第2点,我认为是有必要独立出第一页的,减少运算和字符拼接有利于节省资源的

至于楼主说not in可以解决非主键难于排序查找的问题,我认为这只能做为这种情况的替代,而不是所有情况都要用同一条语句,另外临时字段也是可以解决的

呵呵,写的乱,共同讨论
#7
淡漠的茶2008-04-05 15:14
没说楼主存储过程不好,只是跟帖参加讨论以提醒大家每个项目情况都不同,大家应该根据需求写存储过程,而不是认定一个存储过程就可以通用,一切以性能为目标吗,呵呵

我认为任何开发中都应该有个权重的过程,在再开发前就应该分析客户行为,权重分配,不能为了追求通用,追求简单而简化某些代码。

我认为在中最简单的办法未必是最好的办法,而且这样也有坏处的,很多人在开始学习时候因为掌握某种方法,就不去想更多的办法。
#8
淡漠的茶2008-04-05 15:15
我这人比较喜欢参加讨论,不喜欢简单的回答问题,呵呵。新来的,以后多多指教
#9
风风风风2008-04-05 15:48
提示: 作者被禁止或删除 内容自动屏蔽,只有管理员可见
#10
hebingbing2008-04-05 16:38
首先感谢淡漠的茶的见解……
真的很感谢……
其次,我总结一下淡漠的茶的提出的意见:
1.就是应该将第一页分离出来。
2.就是not in会导致将数据where两遍,导致效率下降。
而我的解释是:
1.用了str((@intPageNo-1)*@intPageSize)可以绝对将第一点解决,因为当intPageNo=1时,(@intPageNo-1)*@intPageSize=0,则select top 0……,相当于不执行,在我的意识里分离出第一页用if再将查询的代码重写一遍,光从麻烦方面来说我都宁愿选择像我那么做,再就是我觉得吧,分离出来和top 0来说吧应该效率上来说是差距不大的甚至可以说是没有。所以第一点我认为是没有问题的。
2.至于第二点嘛,首先说我的吧,我用了not in,这样就避免了计算出当前主键的开始字段和结尾字段,我觉得这样是必要的,因为首先像我上面所说的那样,主键如果不是int型的,用 > < 则不好判断吧,再者,要是要用 > < 判断的话,假如主键是int型的,但是自增标量不是1,或者数据中间并不是连续的。如,将某些字段已经删除了,那么你如何算页开始主键和结束主键?我能想到的办法就是,先求出页开始和结束主键分别是总记录的第几条记录m,n。然后还不是要select数据两次求出第m条和第n条数据的主键值是多少。这样完了再 > < 不就成where数据库三次了吗?那样不是更不值得我们去做,你说了?当然这只是我看的网上的一些代码和我能想到的办法,如果淡漠的茶有什么更高明的办法求出页开始主键和结束主键的值,那么很乐意接受而放弃这种where两次的方法。再说了,如果是主键是字符串型的话……

……淡漠的茶……等你再次提出疑问……

[[it] 本帖最后由 hebingbing 于 2008-4-5 16:42 编辑 [/it]]
#11
淡漠的茶2008-04-05 18:53
先回答风的问题
这个存储过程还有好处的
我们知道分页存储过程一般会采用一个字段作为分页的排除字段,一个或多个字段作为排序字段
目前网上很多分页存储过程会范一个毛病,就是排除字段和排序字段作为同一个参数传进来,这样就引发几个矛盾
1。由于排除字段的局限,无法多个字段排序
2。当这个字段为非主键时,就会发现一个问题,非主键代表不强制表内唯一,这就代表此字段可能有相同值,如果使用not in,就会把多条相同值排除掉,假如第一页中某一行和第二页中某一行上这个字段值相同,比如一个标记类别的非主键字段,都为数码相机,那么我们将第一页作为条件,使用not in来排除的话,那么第二页这行也会被排除,因为not in ,就是不等于这个序列中任何一个,而且后面都会被排除,这就造成丢失
3.当这个字段为非主键时,如果采用> < between and这样的排除方法时,也可能产生丢失,如果是连续几行这个字段值相同,就可能出现多取几行或者少取几行这样的不确定性,比如如果第二页10行,最后三行此字段相同,那么就会有7行 8行 ,如果第三页第一行还相同,那就可能第二页取到11行,这取决与你的表达式,总之是没有一个确切表达式

一般解决办法有三个,
第一种就是lz所用方法了,就是排除与排序参数分离,利用主键唯一性,使用not in将这页前的字段值全排除。这样通过重新排序和top选择,就会得到这页数据了,这是解决的方法之一,具备很大的通用性,对于中小型应用还是合适的
但是这样方法不是没有缺点的,就是在百万数据这个级别上 not in过于缓慢,一个原因是not in对索引不起作用(这点我不太确定,我只是听以前我们经理说的,呵呵),第二个原因由于sql执行顺序问题,导致必然的两次全表搜索,百万级数据上全表检索是很恐怖的,第三个原因not in所用的排除序列,在前面有百万条数据时,这个序列是很长的,可能有百万个值用逗号分隔,这个可能光内存就耗不少吧
 
第二种方法是查询时加入一个临时排序字段,用的不多

第三种方法是分离条件,将主键做一条件,非主键做一条件,或者根据需要就不利用非主键,再某些简单情况下
刚才我合了一个,写的不完全,大家补下吧

CREATE PROCEDURE List
@strFrom varchar(50) ,   --表名
@strFields varchar(200) = '*',  --查询列名
@strWhere  varchar(300) = '',     --查询条件
@strOrder varchar(100) = '', --排序列名
@pageIndex int = 1 ,         --页码
@pageSize int = 15,         --页面大小
@intOrder int = 0,        --排序类型  1为升序
@pk varchar(200)='',     --not in情况下给这个主键值
@countAll bigint output              --返回纪录总数用于计算页面数  
 AS
begin
--声明开始
declare   @countSQL   nvarchar(500) --记数语句
declare @strTemp varchar(100) --临时变量
declare @strOrders varchar(50) --排序语句
declare @strSQL varchar(1000)--主语句
declare   @R bigint
--声明结束

--统计记录总数开始
if @strWhere=''
    begin
        set @countSQL=N'select @R=count(*) from '+convert(nvarchar(50),@strFrom)
    end
else
    begin
        set @countSQL=N'select @R=count(*) from '+convert(nvarchar(50),@strFrom)+' where '+convert(nvarchar(300),@strWhere)
    end
exec SP_EXECUTESQL @countSQL, N' @R BIGINT OUTPUT',  @R OUTPUT
set   @countAll=  @R   
--统计记录总数结束

--排序开始
if @intOrder=0
    begin
        set @strTemp = '<(select min'
        set @strOrders = 'order by  '+@strOrder+' desc '
    end
else
   begin
       set @strTemp = '>(select max'
       set @strOrders = 'order by  '+@strOrder+' asc '
   end
--排序结束

--获取主语句开始
if @pageIndex=1
    begin
        if @strWhere=''
        begin
            set @strSQL='select top '+str(@pageSize)+'  '+@strFields+' from '+@strFrom+' '+@strOrders
        end
        else
        begin
            set @strSQL='select top '+str(@pageSize)+'  '+@strFields+' from '+@strFrom+' where '+@strWhere+' '+@strOrders
        end
    end
else
    begin
        if @pk=''
        begin
        if @strWhere =''
        begin
             set @strSQL= 'select top '+str(@pageSize)+' '+@strFields+' from '+@strFrom+' where '+@strOrder+@strTemp+' ('+@strOrder+')' +' from (select top '+str((@pageIndex-1)*@pageSize)+' '+@strFields+' from '+@strFrom+ ' '+@strOrders+ ') as tempTable ) '+@strOrders
        end
        else
        begin
             set @strSQL= 'select top '+str(@pageSize)+' '+@strFields+' from '+@strFrom+' where '+@strOrder+@strTemp+' ('+@strOrder+')' +' from (select top '+str((@pageIndex-1)*@pageSize)+' '+@strFields+' from '+@strFrom+ ' where '+@strWhere+' '+@strOrders+ ') as tempTable ) AND '+@strWhere+' '+@strOrders
        end
        end
        else
        begin
            if @strWhere=''
            begin
                set @strSQL= 'select top '+str(@pageSize)+' '+@strFields+' from '+@strFrom+' where '+@pk+' not in (select top '+str((@pageIndex-1)*@pageSize)+' '+@pk+' from '+@strFrom+ ' '+@strOrders+ ') '+@strOrders
            end
            else
            begin
                set @strSQL= 'select top '+str(@pageSize)+' '+@strFields+' from '+@strFrom+' where '+@pk+' not in (select top '+str((@pageIndex-1)*@pageSize)+' '+@pk+' from '+@strFrom+' where '+@strWhere+ ' '+@strOrders+ ') and '+@strWhere+' '+@strOrders
            end
        end
    end
--获取主语句结束

exec(@strSQL) --执行
end
GO

写的不太好,大家一起完善吧。可以根据情况减的
楼主,咱第一个问题就不讨论?

[[it] 本帖最后由 淡漠的茶 于 2008-4-5 18:58 编辑 [/it]]
#12
风风风风2008-04-05 19:19
提示: 作者被禁止或删除 内容自动屏蔽,只有管理员可见
#13
hebingbing2008-04-05 20:12
我先从功能上分析一下淡漠的茶的这段代码:
首先淡漠的茶的这段代码可以说比我的多考虑了一个情况.就是在没有主键的情况下,我们先给表排序,完了再开始进行……,其他的方面基本上就差不多了……
再者我从性能上说说:
如果将我和他谈论的第一个问题作为比较的条件的话,他的是比我的多出来了一个将第一页分离的情况。虽然他给出了没有主键的情况的代码,但是这个不能作为性能方面的比较,首先很少有表没有主键的吧,甚至可以说是没有吧,再者……(这个再者留到下一段说)
我们来对比核心的内容:要执行的sql语句……但是好像没有区别……他还是采取了我的那种思想,而且要执行的过程也是没有减少……
总结的说就是如果不将第一个问题做为比较的条件的话,这两个存储过程在性能方面就没有什么区别可言……
而分有主键和没有主键我认为说不上多余反正也用途不是太大,因为很少有表是没有主键的……
不过我还是要感谢淡漠的茶一起探讨

[[it] 本帖最后由 hebingbing 于 2008-4-5 20:18 编辑 [/it]]
#14
ilovetea2008-04-06 22:08
精彩的交流,看了有收获,呵呵
#15
狂砍程序2008-04-06 23:31
精彩 大家加这个群 58808531
  里面都是我们这个论坛的朋友,希望大家都加入
#16
myyetao2008-08-11 16:53
怎么加了where不能用
@SelectWhere=''里面是要加where的吧,看了你的句子有2个句子前面没有写where.
还有where写了后还是出错,找不出什么原因.
你的加了where能用?
#17
masterpi2008-08-25 11:09
1111
#18
tiray2008-08-27 11:57
佩服佩服!楼主写得好,淡漠的茶回帖更好!
#19
高阁逆风2008-08-27 17:03
<div class="notice" style="width: 500px">提示: <em>作者被禁止或删除 内容自动屏蔽</em></div>
#20
高阁逆风2008-08-27 17:03
"<div class="notice" style="width: 500px">提示: <em>作者被禁止或删除 内容自动屏蔽</em></div>"
#21
chenyou04032008-09-09 12:57
又问题多多讨论
我是新手,  呵呵呵 希望以后各位大虾多多帮忙.....
#22
fisherdong20012008-09-20 02:51
where那块有问题
我在VS2005下使用了楼主的存储过程,where那块有问题,我改成如下形式:
set nocount off--打开计数
if @SelectWhere != ''
begin
set @TmpSelect ='select top '+str(@intPageSize)+' '+@TableList+' from '+@TableName+' where '+@SelectOrderId+' not in(select top '+str((@intPageNo-1)*@intPageSize)+' '+@SelectOrderId+' from '+
@TableName+' where '+
@SelectWhere+' '+@SelectOrder+')'+'
and '+@SelectWhere+' '+@SelectOrder
end
else
begin
set @TmpSelect = 'select top '+str(@intPageSize)+' '+@TableList+' from '+@TableName+' where '+@SelectOrderId+' not in(select top '+str((@intPageNo-1)*@intPageSize)+' '+@SelectOrderId+' from '+@TableName+' '+@SelectOrder+') '+@SelectOrder
end
execute sp_executesql @TmpSelect
return(@@rowcount)
GO

在单步调试时却总是提示@SelectWhere,比如@SelectWhere=“IsAdmin=1”的‘=’附近有语法错误,但是却搜索出了正确结果(程序不能通过),请问楼主这是
怎么回事?
#23
初学Delphi2008-09-20 09:19
楼主这种乐于助人的风格值得我们很好学习,凌晨了还给大家找代码。
但事这个存储过程绝对的有问题,我一开始来公司用的就是这个,后来再使用过程中发现问题。
set @TmpSelect = 'select top '+str(@intPageSize)+' '+@TableList+' from '+@TableName+' where '+@SelectOrderId+' not in(select top '+str((@intPageNo-1)*@intPageSize)+' '+@SelectOrderId+' from '+@TableName+' '+@SelectWhere +' '+@SelectOrder+') and '+@SelectWhere +' '+@SelectOrder
这句@SelectOrderId 如果不是主键会出现什么情况?
我们写语句 求表的M到N列  一般写select top N-M * from Tablename where Columns1 not in (select top M Columns1 from Tablename)
这样写必须保证Columns事主键,要不就没意义了,说的极端一点,假如说Columns1在这个表里就一个值,但表有1000行,这样写的结果是空。
  而且执行速度也不行,用Sql自带的分页的存储过程  速度很快。



哈哈  很不好意思 补充一点:我没注意定义的时候楼主写的@SelectOrderId是主键,但我觉得  你这样限制这个分页存储过程就没什么意义了,因为出来一张表(指的@TableName),主键只能是一个字段,对一个字段排序好像用户使用起来……满足不了用户使用。

[[it] 本帖最后由 初学Delphi 于 2008-9-20 09:31 编辑 [/it]]
#24
panxuguang2009-09-18 09:03
过来看看
#25
panxuguang2009-09-18 10:34
--楼主的存储过程有问题
 --我现在帮你改了下,应该没问题了
  --楼主原来是少了两个where
--set @TmpSelect = 'select @RecordCount = count(*) from '+@TableName+' where '+@SelectWhere +''
--select top '+str((@intPageNo-1)*@intPageSize)+' '+@SelectOrderId+' from '+@TableName+' where '+@SelectWhere +' '+@SelectOrder+'



alter proc Proc_getPager
@TableList Varchar(200)='*',--搜索表的字段,比如:’id,datatime,job‘,用逗号隔开
@TableName Varchar(30), --搜索的表名
@SelectWhere Varchar(500)='',--搜索条件,这里不用写where,
@SelectOrderId Varchar(30),--表主键字段名。比如:id
@SelectOrder Varchar(200)='', --排序,可以使用多字段排序但主键字段必需在最前面.也可以不写,比如:order by class asc
@intPageNo int=1, --页号
@intPageSize int=10 ,--每页显示数
@RecordCount int OUTPUT  --总记录数(存储过程输出参数)
as  
   
declare @TmpSelect      NVarchar(600)  
declare @Tmp     NVarchar(600)  

set nocount on--关闭计数

set @TmpSelect = 'select @RecordCount = count(*) from '+@TableName+' where '+@SelectWhere +''

execute sp_executesql
@TmpSelect,    --执行上面的sql语句
N'@RecordCount int OUTPUT' ,   --执行输出数据的sql语句,output出总记录数
@RecordCount  OUTPUT

  if (@RecordCount = 0)    --如果没有贴子,则返回零
       return 0
      
   /*判断页数是否正确*/
  if (@intPageNo - 1) * @intPageSize > @RecordCount   --页号大于总页数,返回错误
     return (-1)
set nocount off--打开计数
if @SelectWhere != ''
begin
set @TmpSelect = 'select top '+str(@intPageSize)+' '+@TableList+' from '+@TableName+' where '+@SelectOrderId+' not in(select top '+str((@intPageNo-1)*@intPageSize)+' '+@SelectOrderId+' from '+@TableName+' where '+@SelectWhere +' '+@SelectOrder+') and '+@SelectWhere +''+@SelectOrder
end
else
begin
set @TmpSelect = 'select top '+str(@intPageSize)+' '+@TableList+' from '+@TableName+' where '+@SelectOrderId+' not in(select top '+str((@intPageNo-1)*@intPageSize)+' '+@SelectOrderId+' from '+@TableName+' '+@SelectOrder+') '+@SelectOrder
end
execute sp_executesql @TmpSelect
return(@@rowcount)
GO

不过不知道楼主如果有多张表的话,能不能实现?

[ 本帖最后由 panxuguang 于 2009-9-18 10:41 编辑 ]
#26
a879577392009-09-19 15:05
    谢谢``   收获胜多!~~
#27
haizeng2009-12-21 10:02
存储过程 很 重要啊!
#28
云霆2010-05-28 11:12
我运行的是后怎么条件不能添加了???
#29
yw54152010-06-10 10:47
我也是想加个classID=""的条件选择就不行了。。。
#30
suyongtao2011-03-25 12:03
不错不错。。。。。。我喜欢!
#31
杨清华2012-07-16 17:00
还是不行啊,我用了LZ和25楼的方法 都出错了
1