在嵌套表中访问嵌套表的内部元素

时间:2015-04-20 13:23:59

标签: oracle plsql oracle10g user-defined-types

我有这段代码

set serveroutput on

declare
  contador pls_integer;
begin
  select count(*) into contador from user_objects where object_name=upper('Matriculación');
  if contador = 1 then
    execute immediate 'drop table Matriculación';
  end if;
  select count(*) into contador from user_objects where object_name=upper('Matrícula');
  if contador = 1 then
    execute immediate 'drop table Matrícula';
  end if;
  select count(*) into contador from user_objects where object_name=upper('AlumnoObj');
  if contador = 1 then
    execute immediate 'drop type AlumnoObj';
  end if;
  select count(*) into contador from user_objects where object_name=upper('TitulaciónObj');
  if contador = 1 then
    execute immediate 'drop type TitulaciónObj';
  end if;
  select count(*) into contador from user_objects where object_name=upper('TablaAsignatura');
  if contador = 1 then
    execute immediate 'drop type TablaAsignatura';
  end if;
  select count(*) into contador from user_objects where object_name=upper('AsignaturaObj');
  if contador = 1 then
    execute immediate 'drop type AsignaturaObj';
  end if;
end;
/

CREATE TYPE AsignaturaObj AS OBJECT(
  idAsignatura    varchar2(6),
  nombre          varchar2(50),
  curso           number(1,0), -- en el original curso y cuatrimestre son varchar2, lo he cambiado por probar
  cuatrimestre    number(1,0),
  créditos        number(3,1),
  coste           number(6,2), -- no necesitamos un número de 16 dígitos como en el original
  idProfesor      varchar2(4)
);
/

CREATE OR REPLACE TYPE TablaAsignatura AS TABLE OF AsignaturaObj;
/

show errors
CREATE TYPE TitulaciónObj AS OBJECT(
  idTitulación    varchar2(6),
  nombre          varchar2(30),
  asignaturas     TablaAsignatura
) ;--NESTED TABLE asignaturas STORE AS Asignatura_Tabla return as value; -- esto NO
/

CREATE TYPE AlumnoObj AS OBJECT(
  dni             varchar2(9),
  idAlumno        varchar2(7),
  nombre          varchar2(20),
  apellido        varchar2(30),
  dirección       varchar2(35), -- calle {coma espacio} número --> 30+2+3=35
  ciudad          varchar2(20),
  teléfono        varchar2(9)
);
/

-- El objetivo es guardar una tabla que contenga por cada alumno cada titulación en la que esté, y por cada titulación las asignaturas en las que está matriculado
-- Nota: si un alumno está matriculado en asignaturas de varias titulaciones habrá una fila por cada titulación
CREATE TABLE Matriculación (
  num_matrícula   number(5,0)     CONSTRAINT Matriculación_PK PRIMARY KEY,
  alumno          AlumnoObj       CONSTRAINT Matriculación_ALUMNO_NN NOT NULL,
  titulación      TitulaciónObj
) NESTED TABLE titulación.asignaturas STORE AS Asignatura_Tabla;

set serveroutput on

declare
  alumno alumnoObj := alumnoObj('12345678A', 'A000000', 'Juan', 'Pérez Quintanilla', 'Calle Quintana, 23', 'Madrid', '913217712');

  asignatura1 asignaturaObj := asignaturaObj('222222', 'Pulpos', 2, 1, '4,5', 120, 'Q000');
  asignatura2 asignaturaObj := asignaturaObj('222223', 'Pólipos', 2, 1, '3', 80, 'Q000');
  titulación titulaciónObj := titulaciónObj('111111', 'Ciencias del Mar', TablaAsignatura(asignatura1, asignatura2));

begin
  insert into matriculación values (contador, alumno, titulación );
end;
/

它创建了一个包含几个对象的表“Matriculación”(注册),其中一个是“TitulaciónObj”类型的对象“titulación”(大学学位)。该对象包含一个“AsignaturaObj”类型的对象表(主题对象)。

我的问题是,在为“Matriculación”表创建一行后,如何访问度数对象(TitulaciónObj)及其内部主题(asignaturas)?

这样的东西
select num_matrícula, alumno, titu.* from matriculación m,  table(m.titulación) titu;

引发ORA-22905:“无法访问非嵌套表项的行”

非常感谢提前

1 个答案:

答案 0 :(得分:3)

使用TABLE()而不是包含对象时,需要引用嵌套表:

SELECT m.num_matrícula,
       m.alumno,
       m.titulación.idTitulación,
       m.titulación.nombre,
       a.*
FROM   matriculación m,
       TABLE(m.titulación.asignaturas) a;