从SEQUENCE跳号说起

http://blog.csdn.net/agaric717/article/details/6690890

一个应用上线后发现一个使用SEQUENCE值来生成的主键经常出现断号,而且断号不是一两个,而是每次几十个几十个的跳。而一般能想到的导致这种跳号的基本就是事务ROLLBAK了导致取出的SEQUENCE没有被插入到表中去,或者就是数据库关闭或者宕掉,导致已经CACHE的SEQUENCE的值丢失,导致的跳号。而在这个场景下,这两种情况都被一一排除了。那么到底什么原因导致的SEQUENCE跳号呢?

一般为了提高从SEQUENCE取值的效率,我们都会对SEQUENCE设置CACHE,而且取值越频繁的,CACHE值设置的越大。那么这个CACHE的值是保存在SHARED POOL中的,而且这块内存是可以被其他程序共享的,也就是说SEQUENCE的CACHE是会在一定的情况下被刷出SHARED POOL的,那么就相当于已经CACHE的SEQUENCE的值就丢失了,这就是导致跳号的原因。下面我们通过一个示例来证明并展开来看看。

SQL> create sequence seq_bear cache 100;

Sequence created
首先创建一个CACHE值为100的SEQUENCE

SQL> select seq_bear.nextval from dual;

NEXTVAL
----------
1
查询SEQUENCE的值为1,再次查询的话,SEQUENCE取下一个值,为2

SQL> /

NEXTVAL
----------
2

SQL> alter system flush shared_pool;

System altered

SQL> select seq_bear.nextval from dual;

NEXTVAL
----------
101
而当我们刷新SHARED_POOL并再次查询的时候,SEQUENCE的值直接跳到了101,说明已经CACHE的值除了被使用的1和2之外,其他的98个全部丢失了。

那么我们如何避免这种情况呢?ORACLE给我们提供了把SEQUENCE KEEP到SHARED POOL中的方法,这样就保证了SEQUENCE的CACHE不会被交换出去,从而避免了这种情况的发生。来看下面的示例:
SQL> select seq_bear.nextval from dual;

NEXTVAL
----------
102
再次查询的时候,值为102
SQL> exec dbms_shared_pool.keep(name => ‘SEQ_BEAR‘,flag => ‘Q‘);

PL/SQL procedure successfully completed
使用DBMS_SHARED_POOL来KEEP SEQUENCE的CACHE

SQL> SELECT SEQ_BEAR.NEXTVAL FROM DUAL;

NEXTVAL
----------
103
再次查询,值为103 
SQL> ALTER SYSTEM FLUSH SHARED_POOL;

System altered

SQL> SELECT SEQ_BEAR.NEXTVAL FROM DUAL;

NEXTVAL
----------
104 
而这时当我们再次刷新SHARED POOL并从SEQUENCE取值的时候,发现SEQUENCE已经不再跳号了。

那么,如何能知道哪些SEQUENCE已经被KEEP了呢?
SQL> select KEPT from v$db_object_cache where name=‘SEQ_BEAR‘;

KEPT
----
YES
通过查询V$DB_OBJECT_CACHE可以看到对象是否已经被KEEP在SHARED POOL,同时,这个视图中还提供了其他的一些信息,比如对象被LOAD了多少次、占用多少内存空间等等信息

DBMS_SHARED_POOL不仅可以KEEP SEQUENCE的CACHE,还可以KEEP住存储过程、包、SQL等等,保证这些不会被交换出SHARED POOL。而KEEP方法使用也很简单,只需要传入要KEEP的对象的名称,如果是其他用户的,则使用USERNAME.OBJECT_NAME作为传入参数;后面一个参数表示要KEEP的对象的类型,比如是包、SEQUENCE、SQL还是其他。具体的取值可以参考ORACLE的文档,上面写的很详细。

同时,DBMS_SHARED_POOL还提供了其他几个方法:

ABORTED_REQUEST_THRESHOLD(threshold_size NUMBER):这个方法可以设定一个界限,保证如果要进入SHARED POOL的对象太大,那么可以设置一个阀值,超过这个阀值的直接报错,而不是经过LRU查找和内存交换之后发现SHARED POOL不够了再报错,可以防止超大对象过度占用SHARED POOL空间。

UNKEEP就是KEEP的反操作

SIZES (minsize NUMBER):这个是列出SHARED POOL中所有大于minsize的对象,对于查找SHARED POOL中大对象并设置合理的ABORTED_REQUEST_THRESHOLD很有用。

总结:
SEQUENCE跳号可能是事务ROLLBAK或者实例被宕过或者是SEQUENCE的CACHE被交换出去过,对于有特殊要求的SEQUENCE或者包、存储过程、触发器等等,可以使用DBMS_SHARED_POOL.KEEP方法,把他们永久保留在SHARED POOL中,从而可以实现一些特定的用途。

时间: 2024-10-12 02:55:36

从SEQUENCE跳号说起的相关文章

ORACLE SEQUENCE跳号总结

在ORACLE数据库中,序列(SEQUENCE)是使用非常频繁的一个数据库对象,但是有时候会遇到序列(SEQUECNE)跳号(skip sequence numbers)的情形,那么在哪些情形下会遇到跳号呢? 事务回滚引起的跳号 不管序列有没有CACHE.事务回滚这种情况下,都会引起序列的跳号.如下实验所示: SQL> create sequence my_sequence   2  start with 1   3  increment by 1   4  maxvalue 99999   5

Sybase自增字段跳号的解决方法

Sybase自增字段跳号原因及影响: 在Sybase数据库中如果数据库在开启的情况下,因为非正常的原因(死机.断电)而导致数据库服务进程强制结束. 那么自动增长的字段将会产生跳号的情况,再往数据表里面插入记录时,自增字段会跳到一个相当大的值上面,以至于主键的自增值极有可能被用尽,或由于数值太大,应用程序中的数据类型如int型等已无法容下这么大的值,而无法在程序里面进行相关的操作. Sybase自增字段跳号解决办法: 如下:sybase在执行shutdown with nowait,再重启后,原来

Oracle Sequence中Cache与NoCache的区别;合适使用

Oracle在创建序列(sequence)时有个参数你可以选择cache或者nocache,下面来讲一下两者的区别: 先来看下创建sequence的语句: create sequence SEQ_ID  minvalue 1  maxvalue 99999999  start with 1  increment by 1  cache n  / nocache  --其中n代表一个整数,默认值为20order; 如果指定CACHE值,Oracle就可以预先在内存里面放置一些Sequence,这样

ORACLE PL/SQL 中序列(sequence)的简易使用方法介绍

如果我是C罗 原文 ORACLE PL/SQL 中序列(sequence)的简易使用方法介绍 sequence在ORACLE中应用十分广泛,就是序列号的意思,会自动增加指定变数,如逐次增加1或者2或者其他. 1.创建序列 Create Sequence 你首先要有CREATE SEQUENCE或者CREATE ANY SEQUENCE 权限 CREATE SEQUENCE CUX_DEMO_SEQUENCEMINVALUE 1MAXVALUE 99999999999START WITH 1000

【转】oracle sequence

原文链接  http://www.cnblogs.com/hyzhou/archive/2012/04/12/2444158.html ORACLE SEQUENCE用法 在oracle中sequence就是序号,每次取的时候它会自动增加.sequence与表没有关系. 1.Create Sequence 首先要有CREATE SEQUENCE或者CREATE ANY SEQUENCE权限. 创建语句如下: 复制代码 CREATE SEQUENCE seqTest INCREMENT BY 1

oracle SEQUENCE 创建, 修改,删除

oracle创建序列化: CREATE SEQUENCE seq_itv_collection            INCREMENT BY 1  -- 每次加几个              START WITH 1399       -- 从1开始计数              NOMAXVALUE        -- 不设置最大值              NOCYCLE               -- 一直累加,不循环              CACHE 10; oracle修改序列

oracle的sequence序列

Sequence:序列是一个数据对象,可以用来生成唯一的整数.是数据库里独立的对象,不依赖任何表,为oracle所特有.用来自动生成主键值,可以用sequence的地方:inert语句的子查询中,update的set等.创建序列 :Create sequence seqname -increament by 1//每次增长1 -start with 1//从1开始增长 -maxvalue 100//最大值是100,nomaxvalue:不设最大值 -nocycle//一直累加没有循环 -noca

Oracle中Sequence使用

Oracle提供了sequence对象,由系统提供自增长的序列号,通常用于生成数据库数据记录的自增长主键或序号的地方. 下面介绍一下关于sequence 的生成,修改,删除等常用的操作: 1. 创建 Sequence 使用如下命令新建sequence(用户需要有CREATE SEQUENCE 或者CREATE ANY SEQUENCE权限): CREATE  SEQUENCE test_sequence  INCREMENT  BY   1   --  每次加的个数据   START  WITH

ORACLE SEQUENCE 介绍

在oracle在sequence序列号被称为,每一次,它会采取增加自己主动,通过序列号需要排序的地方一般使用.  1.Create Sequence  你首先必须CREATE SEQUENCE或CREATE ANY SEQUENCE权限.  CREATE SEQUENCE emp_sequence      INCREMENT BY 1  -- 每次加几个      START WITH 1    -- 从1開始计数      NOMAXVALUE      -- 不设置最大值      NOC