1

我们正在为我们的项目实施版本控制。作为其中的一部分,我们需要检查所有 DB 对象。我们有表、过程、函数、包、视图和物化视图。问题是有很多对象,我们需要明智地放置源代码文件。例如,有表 T1、T2、T3,我们需要文件 Table_T1.txt,其中包含所有对象的 T1 定义(列定义、表的索引和授权)等等。

我知道元数据表,例如DBA_VIEWSdba_sourceDBMS_METADATA.GET_DDL,我可以在其中找到所需的信息,但如何明智地提取该信息对象。目前,我们正在为特定对象获取全部信息,然后将其分离(剪切 - 粘贴)到不同的文件中。有什么聪明的方法来解决这个问题吗?

数据库 - Oracle 10g

4

2 回答 2

0

但是如何明智地提取该信息对象。

正确传递参数。然后,您可以自定义您的输出。

DBMS_METADATA.GET_DDL (object_type, object_name, object_owner)

例如,获取用户 SCOTT 中所有表的元数据

SQL> conn scott/tiger@pdborcl;
Connected.
SQL> set long 200000
SQL> select dbms_metadata.get_ddl('TABLE',t.table_name, 'SCOTT') from US

DBMS_METADATA.GET_DDL('TABLE',T.TABLE_NAME,'SCOTT')
------------------------------------------------------------------------

  CREATE TABLE "SCOTT"."DEPT"
   (    "DEPTNO" NUMBER(2,0),
        "DNAME" VARCHAR2(14),
        "LOC" VARCHAR2(13),
         CONSTRAINT "PK_DEPT" PRIMARY KEY ("DEPTNO")
  USING INDEX PCTFREE 10 INITRANS 2 MAXTRANS 255 COMPUTE STATISTICS
  STORAGE(INITIAL 65536 NEXT 1048576 MINEXTENTS 1 MAXEXTENTS 2147483645
  PCTINCREASE 0 FREELISTS 1 FREELIST GROUPS 1
  BUFFER_POOL DEFAULT FLASH_CACHE DEFAULT CELL_FLASH_CACHE DEFAULT)
  TABLESPACE "USERS"  ENABLE
   ) SEGMENT CREATION IMMEDIATE
  PCTFREE 10 PCTUSED 40 INITRANS 1 MAXTRANS 255
 NOCOMPRESS LOGGING
  STORAGE(INITIAL 65536 NEXT 1048576 MINEXTENTS 1 MAXEXTENTS 2147483645
  PCTINCREASE 0 FREELISTS 1 FREELIST GROUPS 1
  BUFFER_POOL DEFAULT FLASH_CACHE DEFAULT CELL_FLASH_CACHE DEFAULT)
  TABLESPACE "USERS"


  CREATE TABLE "SCOTT"."EMP"
   (    "EMPNO" NUMBER(4,0),
        "ENAME" VARCHAR2(10),
        "JOB" VARCHAR2(9),
        "MGR" NUMBER(4,0),
        "HIREDATE" DATE,
        "SAL" NUMBER(7,2),
        "COMM" NUMBER(7,2),
        "DEPTNO" NUMBER(2,0),
         CONSTRAINT "PK_EMP" PRIMARY KEY ("EMPNO")
  USING INDEX PCTFREE 10 INITRANS 2 MAXTRANS 255 COMPUTE STATISTICS
  STORAGE(INITIAL 65536 NEXT 1048576 MINEXTENTS 1 MAXEXTENTS 2147483645
  PCTINCREASE 0 FREELISTS 1 FREELIST GROUPS 1
  BUFFER_POOL DEFAULT FLASH_CACHE DEFAULT CELL_FLASH_CACHE DEFAULT)
  TABLESPACE "USERS"  ENABLE,
         CONSTRAINT "FK_DEPTNO" FOREIGN KEY ("DEPTNO")
          REFERENCES "SCOTT"."DEPT" ("DEPTNO") ENABLE
   ) SEGMENT CREATION IMMEDIATE
  PCTFREE 10 PCTUSED 40 INITRANS 1 MAXTRANS 255
 NOCOMPRESS LOGGING
  STORAGE(INITIAL 65536 NEXT 1048576 MINEXTENTS 1 MAXEXTENTS 2147483645
  PCTINCREASE 0 FREELISTS 1 FREELIST GROUPS 1
  BUFFER_POOL DEFAULT FLASH_CACHE DEFAULT CELL_FLASH_CACHE DEFAULT)
  TABLESPACE "USERS"


  CREATE TABLE "SCOTT"."BONUS"
   (    "ENAME" VARCHAR2(10),
        "JOB" VARCHAR2(9),
        "SAL" NUMBER,
        "COMM" NUMBER
   ) SEGMENT CREATION DEFERRED
  PCTFREE 10 PCTUSED 40 INITRANS 1 MAXTRANS 255
 NOCOMPRESS LOGGING
  TABLESPACE "USERS"


  CREATE TABLE "SCOTT"."SALGRADE"
   (    "GRADE" NUMBER,
        "LOSAL" NUMBER,
        "HISAL" NUMBER
   ) SEGMENT CREATION IMMEDIATE
  PCTFREE 10 PCTUSED 40 INITRANS 1 MAXTRANS 255
 NOCOMPRESS LOGGING
  STORAGE(INITIAL 65536 NEXT 1048576 MINEXTENTS 1 MAXEXTENTS 2147483645
  PCTINCREASE 0 FREELISTS 1 FREELIST GROUPS 1
  BUFFER_POOL DEFAULT FLASH_CACHE DEFAULT CELL_FLASH_CACHE DEFAULT)
  TABLESPACE "USERS"

因此,这为我提供了SCOTT模式中所有表的DDL 。

同样,您可以对所有其他对象(例如 等)执行相同的INDEXES操作ROLES

要在文本文件中获取 DDL,只需使用SPOOL。因此,您只需要针对不同对象类型的单独脚本来假脱机在各自的文本文件中。

于 2015-02-04T06:36:12.450 回答
0

一个简单的事实是,您不能像对待 Java、C# 或其他文件一样对待数据库对象。

有很多原因,我将仅举几例:

文件本地存储在开发人员的 PC 上,他/她所做的更改不会影响其他开发人员。同样,开发人员也不会受到同事所做更改的影响。在数据库中,(通常)情况并非如此,开发人员共享相同的数据库环境,因此提交给数据库的任何更改都会影响其他人。

发布代码更改是使用签入/提交更改等完成的(取决于您使用的源代码控制工具)。此时,开发人员本地目录中的代码被插入到源代码控制存储库中。想要获取最新代码的开发人员需要从源代码管理工具中请求它。在数据库中,更改已经存在并影响其他数据,即使它没有签入到存储库中。

在文件签入期间,源代码控制工具会执行冲突检查,以查看在您修改本地副本期间,同一文件是否被其他开发人员修改和签入。再次在数据库中没有检查这个。如果您从本地 PC 更改程序,同时我使用本地 PC 的代码修改相同的程序,那么我们会覆盖彼此的更改。

代码的构建过程是通过将代码的标签/最新版本放到一个空目录中,然后执行构建-编译来完成的。输出是我们复制和替换现有的二进制文件。我们不在乎以前的情况。在数据库中,我们无法重新创建数据库,因为我们需要维护数据!部署还执行在构建过程中生成的 SQL 脚本。

执行 SQL 脚本(使用 DDL、DCL、DML(用于静态内容)命令)时,您假定环境的当前结构与创建脚本时的结构相匹配。如果没有,那么您的脚本可能会在您尝试添加已经存在的新列时失败。

将 SQL 脚本视为代码并手动生成它们会导致语法错误、数据库依赖项错误、不可重用的脚本,从而使开发、维护和测试这些脚本的任务变得复杂。此外,这些脚本可能运行在与您想运行的环境不同的环境中。

有时版本控制存储库中的脚本与被测试对象的结构不匹配,然后在生产中会发生错误!

还有很多,但我想你明白了。

我发现有效的方法如下:

使用强制版本控制系统对数据库对象强制执行签出/签入操作。这将确保版本控制存储库与签入的代码相匹配,因为它在签入操作中读取对象的元数据,而不是作为手动完成的单独步骤

使用影响分析,利用基线作为比较的一部分来识别冲突并确定更改(在比较源代码控制存储库和数据库之间的对象结构时)是源自开发的真正更改还是源自不同的路径,然后应该跳过它,例如不同的分支或紧急修复。

我写的一篇文章在这里发表欢迎阅读。

于 2015-03-25T11:59:18.153 回答