1

考虑一个场景,其中有两个表“A”和“B”。

表“A”有一个触发器“Ta”[在我加入这个项目之前很久就写好了,因此我完全不知道触发器],它更新了表“B”中名为“colB”的列。

现在,由于我主要使用表“B”并且关心“colB”的获取方式,我不知道触发器“Ta”是否正在更新此列。

所以我的问题是,是否有直接的 oracle 查询/方法来查找一个表中的列是否被另一个表上运行的任何触发器更新?

在此先感谢您对我的教育。

问候 ab

4

3 回答 3

1

Oracle 细粒度依赖跟踪知道使用了哪些列。不幸的是,无法跟踪该依赖项是用于读取还是写入。并且没有默认DBA_DEPENDENCY_COLUMNS视图来查找此信息。

但幸运的是, Rob van Wijk创造了这样的观点。他的博客有更多信息,包括赠款和create view声明,大约在页面的一半。

例子:

drop table a;
drop table b;

create table a(colA number);
create table b(colB number, read_only number, not_used number);

create or replace trigger Ta
after update or insert or delete on a
begin
    update b set colB = read_only;
end;
/

--What triggers are referencing B's columns?
select owner, name, type, referenced_column
from dba_dependency_columns
where referenced_owner = user
    and referenced_name = 'B'
    and type = 'TRIGGER';

OWNER    NAME  TYPE     REFERENCED_COLUMN
-----    ----  ----     -----------------
JHELLER  TA    TRIGGER  COLB
JHELLER  TA    TRIGGER  READ_ONLY

该视图使用了几个未记录的表和一些高级 SQL 功能。这种视图在生产服务器上不是一个好主意。但它可能比任何涉及解析 SQL 的解决方案都准确得多。

于 2013-07-21T17:26:35.193 回答
1
SELECT *
FROM 
    user_sources
WHERE
    type = 'TRIGGER'
AND UPPER(text) LIKE '%UPDATE A%';

但如果查询在两行中,则它将不起作用,例如:

UPDATE
    A
SET
   ...

因为text匹配相应对象中的给定行。

于 2013-07-20T21:18:50.600 回答
0

简单的例子:

create table table_a(
 id number primary key,
 val varchar2( 100 )
);

create table table_b(
 len number
);

insert into table_b values ( 0 );

set define off
create or replace trigger after_table_a
after insert on table_a for each row
begin
   UpDate 
     table_B
       set len = len + length( :new.val );
end;
/

insert into table_a values ( 1, 'Ala ma kota');
insert into table_a values ( 2, 'As to ali pies');
commit;

select * from table_b;

       LEN
----------
        25 



和查询:

select trigger_name, 
       regexp_substr( trigger_body, 'update\s+table_b',1,1,'i') update_command
from (
  select ut.trigger_name, 
         dbms_metadata.GET_DDL('TRIGGER', ut.trigger_name) trigger_body 
  from user_dependencies ud
  join user_triggers ut on ( ud.type = 'TRIGGER' 
                             and ut.trigger_name = ud.name 
                             and ut.table_name <> ud.referenced_name )
  where ud.referenced_name = 'TABLE_B'
)
where regexp_instr( trigger_body, 'update\s+table_b',1,1,0,'i') > 0 ;


TRIGGER_NAME  UPDATE_COMMAND
------------- ------------------ 
AFTER_TABLE_A UpDate
                table_B
于 2013-07-20T22:21:29.330 回答