【Oracle】仿Oracle Sequence的自定义年份Sequence(适合任何数据库)

Oracle的Sequence很好用,自增,不用维护是它最大的特点,但碰到以下情况就比较麻烦了。

系统开发时常被要求业务流水号的形式为年份+流水号,如20100001,每年的流水号都必须重1开始。

上面的需求如通过Sequence实现方法有多种:

方法1:只使用一个Sequence,程序判断年份,需要在年后第一天使用前重置Sequence到最小值。

优点:开发方便,不需修改程序。

缺点:需要人工干预,且要有很强的重置Sequence意识,不能直观反映出上一年Sequence的使用情况。

方法2:使用多个Sequence,每年使用一个Sequence,程序根据年份使用不同的Sequence,或程序只适用一个Sequence名称,通过手工干预更改当前使用Sequence名称。

优点:比方法1相对灵活,可不需修改程序,可反映历年Sequence使用情况。

缺点:同样需要人工干预,且也需要在年后第一天使用前更改程序或更改当前使用Sequence名称。

综上所见,Sequence在这里使用不太正确,鉴于这种情况,我想出了使用表和函数模拟出Sequence的方法。

(以下方法同样可以在其他数据库上根据实际情况修改语法实现,这里以Oracle为例)

首先表记录的是Sequence的使用情况,参考Oracle的Sequence内容,设计如下:

YEAR:年份

MINVALUE:最小值

MAXVALUE:最大值

NEXTNUMBER:当前值

INCREASEBY:增量

CYCLE:是否循环

create table IDSEQUENCE
(
  YEAR       NUMBER not null,
  MINVALUE   NUMBER not null,
  MAXVALUE   NUMBER not null,
  NEXTNUMBER NUMBER not null,
  INCREASEBY NUMBER not null,
  CYCLE      NUMBER not null
)

实现类似NextVal的函数:

create or replace function IDNextval return number is
  PRAGMA AUTONOMOUS_TRANSACTION; --Oracle函数中需要添加此参数才能在函数中执行SQL
  Result Number;
  lYear number;
  cursor idsequence_cursor(nYear in varchar2) is select * from idsequence where year=nYear;
  refidsequence idsequence_cursor%rowtype;
begin
  lYear := to_char(sysdate, 'yyyy');
  Result := -1;
  
  open idsequence_cursor(lYear);
  fetch idsequence_cursor into refidsequence;
  if not idsequence_cursor%notfound then
    Result := refidsequence.nextnumber;
    
    if refidsequence.nextnumber >= refidsequence.minvalue then --当前值大于等于最小值
      if refidsequence.nextnumber >= refidsequence.maxvalue then --当前值大于等于最大值
        if refidsequence.cycle = 1 then --循环则当前值等于最小值
          update idsequence set nextnumber=minvalue where year=lYear;
        else --不循环则当前值为最小值-1,即无效值
          update idsequence set nextnumber=minvalue-1 where year=lYear;
        end if;
      else --当前值大于等于最小值小于最大值
        if refidsequence.nextnumber+refidsequence.increaseby > refidsequence.maxvalue then --当前值加增量大于最大值
          update idsequence set nextnumber=minvalue where year=lYear;
        else  
          update idsequence set nextnumber=nextnumber+reffeesdocidsequence.increaseby where year=lYear;
        end if;
      end if;  
      commit;
    end if;
  end if;
  close idsequence_cursor;
  
  return(Result);
end IDNextval;

 实现类似CurrVal的函数:

create or replace function IDCurrval return number is
  Result Number;
  lYear number;
  lCount number;
begin
  Result := -1;
  lYear := to_char(sysdate, 'yyyy');
  
  select count(1) into lCount from idsequence where year=lYear;
  if lCount > 0 then
    select nextnumber into Result from idsequence where year=lYear;
  end if;

  return(Result);
end IDCurrval;

使用的时候,用select idnextval from dual获取下一值,用select idcurrval from dual获取当前值。

对于文章开头需求的应用,只需要在开发前把年份记录一次性添加够就行了,在年份切换时函数自动会使用相应年份的记录。

上面函数有一点需要注意,当当前值大于最大值,且循环开关没设置(CYCLE)时,当前值置为最小值-1,即无效值。

同样,当年份记录没有初始化时,获取到的当前值是-1。

posted @ 2011-10-28 16:21  泥头  阅读(993)  评论(0编辑  收藏  举报