【问题标题】:Referring a nested table in a relational insert query在关系插入查询中引用嵌套表
【发布时间】:2011-07-29 22:26:54
【问题描述】:

以下是我拥有的对象类型。基本上我有一个人表和一个子表作为人表的嵌套表。

我有一个与子表(嵌套)有 M:N 关系的 School 表。所以我正在创建一个中间表来插入 child_school 数据。

如何创建中间表并插入数据?

create type school_t as object(
    sid number(5,2),
    name varchar(20))
/

create type child_t as object(
    cid number(5,2),
    name varchar(20))
/

create type childtable_t as table of child_t
/

create type person_t as object(
    pid number(5,2),
    name varchar(20),
    child childtable_t)
/

create table person_tab of person_t(
    pid primary key
)nested table child store as child_table
/

create table school_tab of school_t
/

--有一些问题。以下不起作用。

create type school_child_t as object(
    cid ref person_t,
    sid ref school_t)
/

create table school_child_tab of school_child_t(
    cid references person_tab,
    sid references school_tab
)
/

--这就是我想做的事情

create table school_child_tab(
    cid number(5,2) references childtable_t,
    sid number(5,2) references school_tab
)
/

cid 引用应该是嵌套表中的 cid。问题是指它。

【问题讨论】:

  • 为什么 PERSON 有一个 CHILDTABLE_T 的嵌套表?事实上,您的整个数据模型似乎有点混乱。也许您应该解释您要建模的内容,而不是要求我们解释您的代码?
  • @APC:这是我尝试将弱实体理解为实体表并从外部引用它。我添加了一张图片,以便您了解。
  • 但是为什么一个人可以多个孩子呢?如果 CHILD 是 PERSON 的子类型,那么这是 1:1 关系,而不是 !:M。如果 PERSON 的意思是 TEACHER 或 PARENT,那么你的模型在其他方面是错误的。
  • 是的,我认为你是正确的。应该是 1:1。谢谢!

标签: sql oracle nested-table


【解决方案1】:

我看到了你的编辑,我正要告诉你不可能从外部引用嵌套表。

嵌套表在物理上创建为一个不同的表,它与父表分开保存数据:

SQL> SELECT object_name, object_type
  2    FROM all_objects
  3   WHERE created > trunc(sysdate)
  4     AND object_type = 'TABLE';

OBJECT_NAME                    OBJECT_TYPE
------------------------------ -------------------
SCHOOL_TAB                     TABLE
CHILD_TABLE                    TABLE
PERSON_TAB                     TABLE

在这里您可以看到 Oracle 已经创建了一个 CHILD_TABLE 表,但是它对我们是隐藏的,只能由 Oracle 在内部工作:

SQL> select * from child_table;

ORA-22812: cannot reference nested table column's storage table

在这种情况下,我很确定您无法以任何方式引用子表,但令我惊讶的是,这似乎有效(我们无法从 CHILD_TABLE 中选择,但我们可以引用它):

SQL> alter table child_table add constraint pk_child_table primary key (cid);

Table altered

SQL> CREATE TABLE school_child_tab (
  2     cid REFERENCES child_table,
  3     sid REFERENCES school_tab
  4  );

Table created

你可以像这样构建你的插入(我不太喜欢将数据存储为对象,但你可以这样):

SQL> insert into school_tab values (school_t(1, 'school A'));

1 row inserted

SQL> insert into person_tab values (
  2      person_t(1, 'person A', childtable_t(child_t(1, 'child A'))));

1 row inserted

SQL> insert into school_child_tab values (1, 1);

1 row inserted

【讨论】:

  • +1 有没有人在大学作业之外在数据库表中使用过嵌套表?
  • @Vincent:我想引用嵌套表中 child_t 对象中的 cid。不是 Person_t 中的 pid
  • @TonyAndrews - 我遇到了真正的高度专业化用途。 Oracle Spatial 使用嵌套表,它指向实现可能有效的场景。但我同意大多数示例 - 像这样 - 都是虚假的,最好作为普通表实现。
  • @Nipuna:可以参考嵌套表,看我更新的答案
  • @Vincent:我得到了答案。谢谢
【解决方案2】:

我稍微改变了你的数据模型:

SQL> create type school_t as object(
  2      sid number(5,2),
  3      name varchar(20))
  4  /

Type created.

SQL> create type child_t as object(
  2      cid number(5,2),
  3      name varchar(20))
  4  /

Type created.

SQL> create table school_tab of school_t
  2  /

Table created.

SQL> create table child_tab  of child_t
  2  /

Table created.

SQL>

让我们填充嵌套表:

SQL> insert into child_tab
  2      values (111, 'Fred')
  3  /

1 row created.

SQL> insert into child_tab
  2      values (112, 'Ayesha')
  3  /

1 row created.

SQL> insert into child_tab
  2      values (113, 'Aadil')
  3  /

1 row created.

SQL> insert into school_tab
  2      values (222, 'Bash Street')
  3  /

1 row created.

SQL> insert into school_tab
  2      values (223, 'Greyfriars')
  3  /

1 row created.

SQL> 

这是一个嵌套表:

SQL> create type school_child_t as object(
  2      cid ref child_t,
  3      sid ref school_t)
  4  /

Type created.

SQL> create table school_child_tab of school_child_t
  2  /

Table created.

SQL>

我们像这样填充交集表:

SQL> insert into school_child_tab
  2        select cid, sid
  3        from
  4          ( select ref(c) as cid from child_tab c where c.cid = 111 )
  5          , ( select ref(s) as sid from school_tab s where s.sid = 222 )
  6  /

1 row created.

SQL> insert into school_child_tab
  2        select cid, sid
  3        from
  4          ( select ref(c) as cid from child_tab c where c.cid = 112 )
  5          , ( select ref(s) as sid from school_tab s where s.sid = 222 )
  6  /

1 row created.

SQL> insert into school_child_tab
  2        select cid, sid
  3        from
  4          ( select ref(c) as cid from child_tab c where c.cid = 113 )
  5          , ( select ref(s) as sid from school_tab s where s.sid = 222 )
  6  /

1 row created.

SQL> insert into school_child_tab
  2        select cid, sid
  3        from
  4          ( select ref(c) as cid from child_tab c where c.cid = 113 )
  5          , ( select ref(s) as sid from school_tab s where s.sid = 223 )
  6  /

1 row created.

SQL>

查询返回结果

SQL> select c.name as child_name
  2         , s.name as school_name
  3  from     school_child_tab sc
  4              join child_tab c
  5                  on ( ref(c) = sc.cid )
  6              join school_tab s
  7                  on ( ref(s) = sc.sid )
  8  /

CHILD_NAME           SCHOOL_NAME
-------------------- --------------------
Fred                 Bash Street
Ayesha               Bash Street
Aadil                Greyfriars
Aadil                Bash Street

SQL>

当然,这提出了一个问题:如果要使用对象的 REF,是否需要 ID 列?当然,我认为对于 CHILD_T 类型有一个名为 CID 的类型为 NUMBER 的属性,而对于 SCHOOL_CHILD_T 类型有一个具有相同名称但数据类型为 REF 的属性,这是一种误导。

【讨论】:

  • 谢谢,但我想从外部查询中引用嵌套表。我从 Vincents 上次编辑中得到了我想要的。
猜你喜欢
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 2017-08-13
  • 2019-03-30
  • 1970-01-01
  • 1970-01-01
  • 2018-10-23
  • 1970-01-01
相关资源
最近更新 更多