--参考这个处理示例--得到最新编号的函数
create function f_newid()
returns char(7) --编号位数固定,用char的检索效率高于varchar
as
begin
declare @re char(7)
select @re=max(BHID) from 表 
return(
case when @re is null then 'BH00001'
else 'BH'+right('0000'+cast(cast(right(@re,5) as int)+1 as varchar),5) 
end)
end
go--测试的表,表名与函数中的表名对应
create table 表(
BHID char(7) default dbo.f_newid() --设置默认值,自动生成编号
primary key,  --设置成主键,防止编号冲突
-- constraint UNIQUE_BHID_表 unique, --如果用唯一约束,则删除上面的主键约束语句,改用此句
txt1 varchar(10),
txt2 varchar(10),
num  float)
go--插入的调用
declare @newbh char(7)
select @newbh=dbo.f_newid()
insert 表(BHID,txt1,txt2,num) values(@newbh,'aa','bb',1)
select 刚插入的编号值=@newbh/*--插入数据时,就可以不理会编号字段,直接用这样的语句 多用户同时插入时,如果编号重复,就会有错误发生
此时,前台程序拦截错误,如果是违反约束的错误
只需要重新执行插入的语句即可,此时的编号会自动再重新生成
而重新执行插入语句也很方便,因为根本就不需要改语句 假如几个请求同时发生到达,f_newid()同时运行,取了同样id,会插入同样id? 
约束会阻止此插入操作
前台程序拦截错误,客户重新执行插入的语句,有可能造成某客户长期等待危险
如果用表级锁,那客户的等待时间更长,而且影响正常的读表操作.
--*/
go
--删除测试
drop table 表
drop function f_newid

解决方案 »

  1.   

    需要表中有主键字段来配合处理(或者用临时表),因为SQL中是没有记录号(记录顺序的),所以没有主键的话,无法确定记录顺序
      

  2.   

    --用临时表可以这样处理--测试的表
    create table T(a int)
    insert T  select 10
    union all select 5
    union all select 20
    union all select 15
    union all select 22
    union all select 6
    union all select 40
    go--查询的存储过程
    create proc p_qry
    @value int
    as
    set nocount on
    select id=identity(int),a into #t from T
    select a from #t a 
    where isnull((select sum(a) from #t where id<a.id),0)<@value
    go--调用
    exec p_qry 20exec p_qry 50
    go--删除测试
    drop table T
    drop proc p_qry/*--测试结果a           
    ----------- 
    10
    5
    20
    a           
    ----------- 
    10
    5
    20
    15
    --*/
      

  3.   

    to zjcxc(邹建)
    高人!
    谢谢了,虽然我也对t进行了自关联,但是没有取出所要的数据,关键是再ISNULL(,),还是不太明白ISNULL(,)再这里的用法,希望能够详细指点指点,在此多谢了!
      

  4.   

    isnull(<表达式>,<值>)如果<表达式>的结果是null,则返回<值>
    否则返回<表达式>的结果