日期:2014-05-16  浏览次数:21054 次

在Oracle、MySQL、MS SQL Server中创建自动增长字段

好吧,今天面试有道题,要各个数据库怎么建立自增长字段,顺便复习一下吧,最近面试很多数据库问题。。。

一:Oracle

?

Oracle中创建自增长字段,要建序列和触发器, 
1.先创建序列 

通过创建序列来实现 
ORACLE SEQUENCE的简单介绍 


在oracle中sequence就是所谓的序列号,每次取的时候它会自动增加,一般用在需要按序列号排序的地方。 
1、Create Sequence 
你首先要有CREATE SEQUENCE或者CREATE ANY SEQUENCE权限, 
CREATE SEQUENCE emp_sequence 
INCREMENT BY 1 -- 每次加几个 
START WITH 1 -- 从1开始计数 
NOMAXVALUE -- 不设置最大值 
NOCYCLE -- 一直累加,不循环 
CACHE 10; 

一旦定义了emp_sequence,你就可以用CURRVAL,NEXTVAL 
CURRVAL=返回 sequence的当前值 
NEXTVAL=增加sequence的值,然后返回 sequence 值 
比如: 
emp_sequence.CURRVAL 
emp_sequence.NEXTVAL 

可以使用sequence的地方: 
- 不包含子查询、snapshot、VIEW的 SELECT 语句 
- INSERT语句的子查询中 
- NSERT语句的VALUES中 
- UPDATE 的 SET中 

可以看如下例子: 
INSERT INTO emp VALUES 
(empseq.nextval, 'LEWIS', 'CLERK',7902, SYSDATE, 1200, NULL, 20); 

SELECT empseq.currval FROM DUAL; 

但是要注意的是: 
- 第一次NEXTVAL返回的是初始值;随后的NEXTVAL会自动增加你定义的INCREMENT BY值,然后返回增加后的值。CURRVAL 总是返回当前SEQUENCE的值,但是在第一次NEXTVAL初始化之后才能使用CURRVAL,否则会出错。一次NEXTVAL会增加一次SEQUENCE的值,所以如果你在同一个语句里面使用多个NEXTVAL,其值就是不一样的。明白? 

- 如果指定CACHE值,ORACLE就可以预先在内存里面放置一些sequence,这样存取的快些。cache里面的取完后,oracle自动再取一组到cache。 使用cache或许会跳号, 比如数据库突然不正常down掉(shutdown abort),cache中的sequence就会丢失. 所以可以在create sequence的时候用nocache防止这种情况。 

2、Alter Sequence 
你或者是该sequence的owner,或者有ALTER ANY SEQUENCE 权限才能改动sequence. 可以alter除start至以外的所有sequence参数.如果想要改变start值,必须 drop sequence 再 re-create . 
Alter sequence 的例子 
ALTER SEQUENCE emp_sequence 
INCREMENT BY 10 
MAXVALUE 10000 
CYCLE -- 到10000后从头开始 
NOCACHE ; 


影响Sequence的初始化参数: 
SEQUENCE_CACHE_ENTRIES =设置能同时被cache的sequence数目。 

可以很简单的Drop Sequence 
DROP SEQUENCE order_seq;

2.创建触发器 

-- 
CREATE   OR   REPLACE   TRIGGER   USERADD 
BEFORE   INSERT   ON   MEMBERRG 
FOR   EACH   ROW 
BEGIN 
        emp_sequence.NEXTVAL   INTO   :NEW.NUM   FROM   DUAL; 
END; 

?

?但是如果数据库表已经有数据了怎么办呢?又不能重新去插数据触发触发器,这样很繁琐,怎么办?

?

在表中增加一个列,为自增序列,然后执行:

update 表 set 自增序列=rownum;

这样就把以前的3000多万行都编号了,然后再用Sequence做序列;

? ?OK,下来举个具体的例子好了:

?

?

一、在Oracle中的没有自动增长的数据类型,但可以使用序列(Sequence)代替。具体步骤如下:

1、 创建一个t_user的用户表:
create table t_user(
       id number primary key,
       userName varchar2(15) not null unique,
       userPass varchar2(20)
);

2、 创建一个序列seq_user_id:
create sequence seq_user_id
start with 1 –从1开始
increment by 1;--每次增量为1

3、 创建一个触发器使给id 赋值:
create trigger tr_user_id
before
insert
on t_user
for each row
begin
     select seq_user_id.nextval into :new.id from dual;
end;



insert into t_user (userName,userPass) values('handson','handson');

insert into t_user (userName,userPass) values('admin','admin');

?接下来介绍一个通用一点的做法,使用存储过程:

?

二、创建一个存储过程来建立自动增长字段:

1、 Oracle中执行动态SQL时要显示授权(即使该用户拥有该相关权限)
grant create any sequence to scott;

grant create any trigger to scott;


2、建立一个创建自增字段的存储过程
create or replace procedure pro_addIncrement(tableName in varchar2 , columnName in varchar2)

as

strsql varchar2(1000);

begin

strsql := 'create sequence seq_'||tableName||'_'||columnName||' start with 1 increment by 1';

execute immediate strsql;    

strsql := 'create or replace trigger tr_'||tableName||'_'||columnName||' before insert on '||tableName||' for each row

begin

select seq_'||tableName||'_'||columnName||'.nextval into :new.'||columnName||' from dual;

end;';

execute immediate strsql;

end;


2、 调用存储过程建立自增字段并测试
create table t_user(

       id number primary key,

       userName varchar2(20) not null unique,

       userPass varchar2(20)

);

exec pro_addIncrement('t_user','id');

insert into t_user (userName,userPass) values('handson','handson');

insert into t_user (userName,userPass) values('admin','admin');

insert into t_user (userName,userPass) values('yangdongxin','pass');

insert into t_user (userName,userPass) values('java','java');

?

二:mysql

?

?