我正在创建一个包以在我的模式中生成对象的DDL(您解析对象名称,并返回带有DDL的clob
),这样我就可以生成文件,并将它们直接放入SVN。 / p>
我正在使用dbms_metadata.get_ddl
,它适用于除表格/物化视图之外的所有对象。
如果我创建一个表:
create table stackoverflow
( col_1 varchar2(64)
, col_2 number
, col_3 date);
create index idx_test on stackoverflow(col_1);
alter table stackoverflow add constraint ui_test unique (col_2) using index;
使用以下命令生成DDL:
begin
dbms_output.put_line(dbms_metadata.get_ddl( object_type => 'TABLE'
, name => 'STACKOVERFLOW')
);
end;
它给了我们:
CREATE TABLE "TEST_SCHEMA"."STACKOVERFLOW"
( "COL_1" VARCHAR2(64),
"COL_2" NUMBER,
"COL_3" DATE,
CONSTRAINT "UI_TEST" UNIQUE ("COL_2")
USING INDEX PCTFREE 10 INITRANS 2 MAXTRANS 255 COMPUTE STATISTICS
STORAGE(INITIAL 131072 NEXT 131072 MINEXTENTS 1 MAXEXTENTS 2147483645
PCTINCREASE 0 FREELISTS 1 FREELIST GROUPS 1
BUFFER_POOL DEFAULT FLASH_CACHE DEFAULT CELL_FLASH_CACHE DEFAULT)
TABLESPACE "USERS_DATA_TS" ENABLE
) SEGMENT CREATION IMMEDIATE
PCTFREE 10 PCTUSED 40 INITRANS 1 MAXTRANS 255
NOCOMPRESS LOGGING
STORAGE(INITIAL 131072 NEXT 131072 MINEXTENTS 1 MAXEXTENTS 2147483645
PCTINCREASE 0 FREELISTS 1 FREELIST GROUPS 1
BUFFER_POOL DEFAULT FLASH_CACHE DEFAULT CELL_FLASH_CACHE DEFAULT)
TABLESPACE "USERS_DATA_TS"
要获取我们可以使用的所有相关索引:
begin
dbms_output.put_line(dbms_metadata.get_dependent_ddl( object_type => 'INDEX'
, base_object_name => 'STACKOVERFLOW'));
end;
拥有:
CREATE INDEX "TEST_SCHEMA"."IDX_TEST" ON "MF"."STACKOVERFLOW" ("COL_1")
PCTFREE 10 INITRANS 2 MAXTRANS 255 COMPUTE STATISTICS
STORAGE(INITIAL 131072 NEXT 131072 MINEXTENTS 1 MAXEXTENTS 2147483645
PCTINCREASE 0 FREELISTS 1 FREELIST GROUPS 1
BUFFER_POOL DEFAULT FLASH_CACHE DEFAULT CELL_FLASH_CACHE DEFAULT)
TABLESPACE "USERS_DATA_TS"
CREATE UNIQUE INDEX "TEST_SCHEMA"."UI_TEST" ON "MF"."STACKOVERFLOW" ("COL_2")
PCTFREE 10 INITRANS 2 MAXTRANS 255 COMPUTE STATISTICS
STORAGE(INITIAL 131072 NEXT 131072 MINEXTENTS 1 MAXEXTENTS 2147483645
PCTINCREASE 0 FREELISTS 1 FREELIST GROUPS 1
BUFFER_POOL DEFAULT FLASH_CACHE DEFAULT CELL_FLASH_CACHE DEFAULT)
TABLESPACE "USERS_DATA_TS"
我想创建一个文件,其中包含:创建表,约束,索引,授权(要一个文件包含所有需要的定义)并使用dbms_metadata
我看起来不可能。
我输出的问题是:
双重引用名称
DDL中的模式名称使我们很难跨多个模式编译相同的DDL。要解决这个问题,我们需要为正则表达式创建一些或者添加类似下面的内容可以解决这个问题:
dbms_metadata.SET_REMAP_PARAM(dbms_metadata.SESSION_TRANSFORM,'REMAP_SCHEMA','TEST_SCHEMA','');
但你必须增加8行:
hOpenOrig0 := DBMS_METADATA.OPEN('TABLE');
DBMS_METADATA.SET_FILTER(hOpenOrig0,'NAME',p_object_name);
DBMS_METADATA.SET_FILTER(hOpenOrig0,'SCHEMA',get_table.owner);
tr := dbms_metadata.add_transform(hOpenOrig0, 'MODIFY');
hTransDDL := DBMS_METADATA.ADD_TRANSFORM(hOpenOrig0,'DDL');
dbms_metadata.set_remap_param(tr, name => 'REMAP_SCHEMA', old_value => user, new_value => '');
get_package_spec.ddl := DBMS_METADATA.FETCH_CLOB(hOpenOrig0);
DBMS_METADATA.CLOSE(hOpenOrig0);
无法在同一时刻提取约束(使用索引)和索引。由于重复ui_test
的定义,您无法连接输出。是的,可以选择从get_ddl
中删除约束,但我们正在放弃约束/检查。
PL / SQL Developer如何创建输出
-- Create table
create table STACKOVERFLOW
(
col_1 VARCHAR2(64),
col_2 NUMBER,
col_3 DATE
)
tablespace USERS_DATA_TS
pctfree 10
pctused 40
initrans 1
maxtrans 255
storage
(
initial 128K
next 128K
minextents 1
maxextents unlimited
pctincrease 0
);
-- Create/Recreate indexes
create index IDX_TEST on STACKOVERFLOW (COL_1)
tablespace USERS_DATA_TS
pctfree 10
initrans 2
maxtrans 255
storage
(
initial 128K
next 128K
minextents 1
maxextents unlimited
pctincrease 0
);
-- Create/Recreate primary, unique and foreign key constraints
alter table STACKOVERFLOW
add constraint UI_TEST unique (COL_2)
using index
tablespace USERS_DATA_TS
pctfree 10
initrans 2
maxtrans 255
storage
(
initial 128K
next 128K
minextents 1
maxextents unlimited
pctincrease 0
);
有没有人知道创建类似于PL / SQL Developer的输出的方法?我猜他们创建了一些dbms_metadata.get_xml
的XML解析器来创建一个更漂亮的版本(缩进,顺序,一切都在适当的位置,随时可以编译)。
当然我可以玩正则表达式或user_indexes
,但这不是重点。
PS。 DBMS_METADATA.SET_TRANSFORM_PARAM(DBMS_METADATA.SESSION_TRANSFORM,'PRETTY',true);
是一个奇怪的定义是什么“漂亮”。
答案 0 :(得分:7)
dbms_metadata.get_dll将oracle对象作为xml获取,然后通过xslt将其转换为ddl脚本。
有用的表格select table_name from all_tables where table_name like 'META%'
列表。
METASTYLESHEET - 将样式表映射到它的名称
METAXSL $ - 将XMLTAG映射到样式表名称 - 将其链接到第1个表
对于表,oracle使用kutable for xml to ddl for a index oracle使用kuindex ...等等。
通过设置参数,您可以更改转换行为。要找到有用的参数检查METAXSLPARAM $表或在样式表文档中搜索它。 EMIT_SCHEMA - 我在kucommon xslt
中找到了EXECUTE DBMS_METADATA.SET_TRANSFORM_PARAM(DBMS_METADATA.SESSION_TRANSFORM,'EMIT_SCHEMA',false); --undocumented remove schema
EXECUTE DBMS_METADATA.SET_TRANSFORM_PARAM(DBMS_METADATA.SESSION_TRANSFORM,'SEGMENT_CREATION',false); --undocumented remove segement creation
EXECUTE DBMS_METADATA.SET_TRANSFORM_PARAM(DBMS_METADATA.SESSION_TRANSFORM,'CONSTRAINTS_AS_ALTER',true);
select dbms_metadata.get_ddl( object_type => 'TABLE' , name => 'STACKOVERFLOW') from dual;
答案 1 :(得分:-2)
我建议您使用expdp / impdp进行ddl导出。
导出架构:faydin
,带有以下expdp。
expdp userid=faydin/***** directory=ORA_TMP_DIR reuse_dumpfiles=y content=METADATA_ONLY exclude=STATISTICS schemas=faydin dumpfile=metadata.dmp
使用impdp
为ddl.sql
获取user : faydin remapped as: faydin3
的ddl
impdp userid=faydin/***** directory=ORA_TMP_DIR dumpfile=metadata.dmp sqlfile=ddl.sql remap_schema=faydin:faydin3
通过删除impdp命令中的sqlfile=ddl.sql
短语将模式导入db。