在删除序列之前查找序列生成的最后一个数字

时间:2018-11-27 12:36:25

标签: oracle sequence oracle12c database-administration

我们发现客户的UAT实例之一中不存在序列“ _SEQUENCE”。我们不知道如何删除它以及何时删除它。这是非常关键的序列,因为此序列生成的数字在整个数据库的许多表中都用作唯一的列值。换句话说,数据库中任何两个表中的任何两列(特定列类型)都不会具有与值相同的数字。在这些列中的少数列上,我们还具有唯一索引。

我们可以再次创建序列,但是我们不知道序列的初始值应该是什么,因为我们不知道旧序列生成的最后一个数字是什么。如果我们将错误的数字设置为初始值,并且偶然地,如果它为已经存在的表的同一列生成了相同的数字,那么我们可能最终会得到唯一的键冲突异常。

我们可以将初始值设置为一个很大的数字,但这是最后的选择。现在;

  1. 是否可以找到序列“ _SEQUENCE”在删除之前生成的最后一个数字?
  2. 是否可以找到哪个进程删除了序列“ _SEQUENCE”以及何时删除了该序列?

1 个答案:

答案 0 :(得分:0)

闪回操作不适用于序列,而可用于表。可能由DDLSYS数据库用户中主要创建的SYSTEM触发器的贡献产生了一种机制。作为示例,请考虑以下过程:

create or replace procedure pr_ddl_oper as
  v_oty          varchar2(75) := ora_dict_obj_type;
  v_don          varchar2(75) := ora_dict_obj_name;
  v_evt          varchar2(75) := ora_sysevent;
  v_olu          varchar2(75) := nvl(ora_login_user,'Unknown Schema');
  v_sql          ora_name_list_t;
  v_stm          clob;
  v_sct          owa.vc_arr;
  n              pls_integer;
  n_max          pls_integer := 10000; -- max number of rows for CLOB object 
                                       -- to hold object's source.
begin
      v_sct(1) := 'SESSIONID';
      v_sct(2) := 'IP_ADDRESS';
      v_sct(3) := 'TERMINAL';
      v_sct(4) := 'OS_USER';
      v_sct(5) := 'AUTHENTICATION_TYPE';
      v_sct(6) := 'CLIENT_INFO';
      v_sct(7) := 'MODULE';
      for i in 1..7
      loop
       v_sct(i) := sys_context('USERENV',v_sct(i));
      end loop;
      select decode(v_sct(1),0,null,v_sct(1)),decode(upper(v_sct(3)),
            'UNKNOWN',null,v_sct(3)) 
        into v_sct(1),v_sct(3) 
        from dual;

       n := ora_sql_txt( v_sql );
      if n > n_max then
       n := n_max;
      end if;
      for i in 1..n
      loop
       v_stm := v_stm || v_sql(i);
      end loop;

       insert into log_ddl_oper(event_time,usr,evnt,stmt,sessionid,ip,terminal,os_user,
         auth_type,object_type,object_name,client_info,module_info)
       values(sysdate,v_olu,v_evt,v_stm,v_sct(1),v_sct(2),v_sct(3),v_sct(4),v_sct(5),
         v_oty,v_don,v_sct(6),v_sct(7));
end;

可以被此触发器调用以作为将来的参考:

--| Compiling this trigger, especially for Production Systems, should be handled with care |

create or replace trigger system.trg_admin_ddl before ddl on database
declare
begin
 pr_ddl_oper;
end;

通过连接SYS用户,您可以在last_number列中查询已删除的Sequence:

select last_number
  from dba_sequences 
  as of timestamp to_timestamp('2018-11-27 23:50:17', 'YYYY-MM-DD HH24:MI:SS') s 
 where s.sequence_name = 'MYSEQ';

可以检测到时间戳值的位置
select l.event_time 
--> returns to_timestamp('2018-11-27 23:50:17', 'YYYY-MM-DD HH24:MI:SS')
--> to use for the above SQL Select statement
  from log_ddl_oper l
 where l.object_name = 'MYSEQ'
   and l.evnt = 'DROP'