| Oracle9i XML Database Developer's Guide - Oracle XML DB Release 2 (9.2) Part Number A96620-01 | 
 | 
This chapter describes how to create and use XMLType views. It contains the following sections:
XMLType views wrap existing relational and object-relational data in XML formats. The major advantages of using XMLType views are:
XMLSchema functionality without having to migrate your base legacy data.XMLType views, you can experiment with various other forms of storage, besides the object-relational or CLOB storage alternatives available to XMLType tables.XMLType views are similar to object views. Each row of an XMLType view corresponds to an XMLType instance. The object identifier for uniquely identifying each row in the view can be created using an expression such as extract() on the XMLType value.
Similar to XMLType tables, XMLType views can conform to an XML schema. This provides stronger typing and enables optimization of queries over these views.
To use XMLType views with XML schemas, you must first register your XML schema with annotations that represent the bi-directional mapping from XML to SQL object types. An XMLType view conforming to this registered XML schema can then be created by providing an underlying query that constructs instances of the appropriate SQL object type.
This chapter describes the two main ways you can create XMLType views:
You can create a view of XMLType or a view with one or more XMLType columns, by using the SQL XML generation functions, particularly those that comply with the emerging SQLX standards.
| See Also: Chapter 10, "Generating XML Data from the Database", for details on SQLX generation functions. | 
The following statement creates an XMLType view using XMLElement() generation function:
DROP TABLE employees; CREATE TABLE employees (empno number(4), fname varchar2(20), lname varchar2(20), hire date, salary number(6)); INSERT INTO employees VALUES (2100, 'John', 'Smith', Date'2000-05-24', 30000); INSERT INTO employees VALUES (2200, 'Mary', 'Martin', Date'1996-02-01', 30000); CREATE OR REPLACE VIEW Emp_view OF XMLTYPE WITH OBJECT ID (EXTRACT(sys_nc_rowinfo$,'/Emp/@empno').getnumberval()) AS SELECT XMLELEMENT("Emp", XMLAttributes(empno), XMLForest(e.fname ||' '|| e.lname AS "name", e.hire AS "hiredate")) AS "result" FROM employees e WHERE salary > 20000;
A query against the XMLType view returns the following employee data in XML format:
SELECT * FROM Emp_view; <Emp empno="2100"> <name>John Smith</name> <hiredate>2000-05-24</hiredate> </Emp> <Emp empno="2200"> <name>Mary Martin</name> <hiredate>1996-02-01</hiredate> </Emp>
empno attribute in the document should become the unique identifier for each row. SYS_NC_ROWINFO$ is a virtual column that references the row XMLType instance.
You can perform DML operations on these XMLType views, but, in general, you must write instead-of triggers to handle the DML operation.
XMLType Views can also be created using SYS_XMLGEN. An equivalent query that produces the same query results using SYS_XMLGEN is as follows :
CREATE TYPE Emp_t AS OBJECT ("@empno" number(4), fname varchar2(2000), lname varchar2(2000), hiredate date); CREATE VIEW employee_view OF XMLTYPE WITH OBJECT ID (EXTRACT(sys_nc_rowinfo$,'/Emp/@empno').getnumberval()) AS SELECT SYS_XMLGEN(emp_t(empno, fname, lname, hire), XMLFORMAT('EMP')) FROM employees e WHERE salary > 20000;
Existing data in relational or object-relational tables or views can be exposed as XML using this mechanism. In addition, queries involving simple XPath traversal over SYS_XMLGEN views are candidates for query rewrite to directly access the object attributes.
To wrap relational data with strongly-typed XML, that is, XML based on an XML schema, perform these steps:
XMLType view and specify the XML schema URL and the root element name. The underlying view query first constructs the object instances and then converts them to XML. This step can also be done in two steps: 
The mechanism for creating XMLType views is more convenient when you already have an object-relational schema and want to map it directly to XML. Also, since the view is based on XML schema, it derives several performance (memory and access) optimizations.
You can create XML schema-based XMLType views without creating object types. For this, you can use the SQL XML generation functions or transformation functions to generate an XML schema conformant XMLType instance. The use of object types with schemas however, enables Query Rewrite functionality.
Consider the following examples based on the canonical employee -department relational tables and XML views of this data:
For the first example view, to wrap the relational employee data with nested department information as XML, follow these steps:
CREATE OR REPLACE TYPE dept_t AS OBJECT ( DEPTNO NUMBER(2), DNAME VARCHAR2(14), LOC VARCHAR2(13) ); / CREATE OR REPLACE TYPE emp_t AS OBJECT ( EMPNO NUMBER(4), ENAME VARCHAR2(10), JOB VARCHAR2(9), MGR NUMBER(4), HIREDATE DATE, SAL NUMBER(7,2), COMM NUMBER(7,2), DEPT DEPT_T ); /
You can create the XML schema by hand or you can use the DBMS_XMLSchema package to generate the XML schema automatically from the existing object types. For example:
SELECT DBMS_XMLSchema.generateSchema('SCOTT','EMP_T') AS result FROM DUAL;
generates the XML schema for the employee type. You can supply various arguments to this function to add namespaces, and so on. You can further edit the XML schema to change the various default mappings that were generated. generateSchemas() function in the package generates a list of XML schemas one for each different SQL database schema referenced by the object type and its attributes.
XML schema, emp.xsd also specifies how the XML elements and attributes are mapped to their corresponding attributes in the object types, as follows:
BEGIN dbms_xmlschema.deleteSchema('http://www.oracle.com/emp.xsd', 4); END; / BEGIN dbms_xmlschema.registerSchema('http://www.oracle.com/emp.xsd', '<schema xmlns="http://www.w3.org/2001/XMLSchema" targetNamespace="http://www.oracle.com/emp.xsd" version="1.0" xmlns:xdb="http://xmlns.oracle.com/xdb" elementFormDefault="qualified"> <element name = "Employee" xdb:SQLType="EMP_T" xdb:SQLSchema="SCOTT"> <complexType> <sequence> <element name = "EmployeeId" type = "positiveInteger" xdb:SQLName="EMPNO" xdb:SQLType="NUMBER"/> <element name = "Name" type = "string" xdb:SQLName="ENAME" xdb:SQLType="VARCHAR2"/> <element name = "Job" type = "string" xdb:SQLName="JOB" xdb:SQLType="VARCHAR2"/> <element name = "Manager" type = "positiveInteger" xdb:SQLName="MGR" xdb:SQLType="NUMBER"/> <element name = "HireDate" type = "date" xdb:SQLName="HIREDATE" xdb:SQLType="DATE"/> <element name = "Salary" type = "positiveInteger" xdb:SQLName="SAL" xdb:SQLType="NUMBER"/> <element name = "Commission" type = "positiveInteger" xdb:SQLName="COMM" xdb:SQLType="NUMBER"/> <element name = "Dept" xdb:SQLName="DEPT" xdb:SQLType="DEPT_T" xdb:SQLSchema="SCOTT"> <complexType> <sequence> <element name = "DeptNo" type = "positiveInteger" xdb:SQLName="DEPTNO" xdb:SQLType="NUMBER"/> <element name = "DeptName" type = "string" xdb:SQLName="DNAME" xdb:SQLType="VARCHAR2"/> <element name = "Location" type = "string" xdb:SQLName="LOC" xdb:SQLType="VARCHAR2"/> </sequence> </complexType> </element> </sequence> </complexType> </element> </schema>', TRUE, FALSE, FALSE); END; /
The preceding statement registers the XML schema with the target location:
"http://www.oracle.com/emp.xsd"
With the one-step process you must create an XMLType view on the relational tables as follows:
CREATE OR REPLACE VIEW emp_xml OF XMLTYPE XMLSCHEMA "http://www.oracle.com/emp.xsd" ELEMENT "Employee" WITH OBJECT ID (ExtractValue(sys_nc_rowinfo$, '/Employee/EmployeeId')) AS SELECT emp_t(e.empno, e.ename, e.job, e.mgr, e.hiredate, e.sal, e.comm, dept_t(d.deptno, d.dname, d.loc)) FROM emp e, dept d WHERE e.deptno = d.deptno;
This example uses the extractValue() SQL function here in the OBJECT ID clause, since extractValue() can automatically figure out the appropriate SQL datatype mapping (in this case a SQL Number) using the XML schema information.
In the two step process, you first create an object-relational view, then create an XMLType view on the object-relational view, as follows:
CREATE OR REPLACE VIEW emp_v OF emp_t WITH OBJECT ID (empno) AS SELECT emp_t(e.empno, e.ename, e.job, e.mgr, e.hiredate, e.sal, e.comm, dept_t(d.deptno, d.dname, d.loc)) FROM emp e, dept d WHERE e.deptno = d.deptno; -- Create the employee XMLType view over the emp_v object view CREATE OR REPLACE VIEW emp_xml OF XMLTYPE XMLSCHEMA "http://www.oracle.com/emp.xsd" ELEMENT "Employee" WITH OBJECT ID DEFAULT AS SELECT VALUE(p) FROM emp_v p;
You can also create the XMLType views using the SQL XML generation functions without the need for object types. You can also use XMLTransform() or other SQL functions which generate XML.The resultant XML must be conformant to the XML schema specified for the view.
With the one-step process you must create an XMLType view on the relational tables without having to create and register any object type as follows:
CREATE OR REPLACE VIEW emp_xml OF XMLTYPE XMLSCHEMA "http://www.oracle.com/emp.xsd" ELEMENT "Employee" WITH OBJECT ID (extract(sys_nc_rowinfo$, '/Employee/EmployeeId/text()').getnumberval()) AS SELECT XMLElement("Employee", XMLAttributes( 'http://www.oracle.com/emp.xsd' AS "xmlns" , 'http://www.w3.org/2001/XMLSchema-instance' AS "xmlns:xsi", 'http://www.oracle.com/emp.xsd http://www.oracle.com/emp.xsd' AS "xsi:schemaLocation"), XMLForest(e.empno AS "EmployeeId", e.ename AS "Name", e.job AS "Job" , e.mgr AS "Manager", e.hiredate AS "HireDate", e.sal AS "Salary", e.comm AS "Commission", XMLForest(d.deptno AS "DeptNo", d.dname AS "DeptName", d.loc AS "Location") AS "Dept")) FROM emp e, dept d WHERE e.deptno = d.deptno;
The XMLElement() function creates the Employee XML element and the inner XMLForest() creates the kids of the employee element. The XMLAttributes clause inside the XMLElement() constructs the required XML namespace and schema location attributes so that the XML generated conforms to the view's XML schema.The innermost XMLForest() function creates the department XML element that is nested inside the Employee element.
The XML generation function simply generate a non-XML schema-based XML instance. However, in the case of XMLType views, as long as the names of the elements and attributes match those in the XML schema, Oracle converts this XML implicitly into a well-formed and valid XML schema-based document.
If you have complicated XML schemas involving multiple namespaces, then you would need to use the partially escaped mapping provided in the SQL functions and create elements with the appropriate namespaces and prefixes.
-- For example the SQL query: SELECT XMLElement("ipo:Employee", XMLAttributes('http://www.oracle.com/emp.xsd' AS "xmlns:ipo", 'http://www.oracle.com/emp.xsd http://www.oracle.com/emp.xsd' AS "xmlns:xsi"), XMLForest(e.empno AS "ipo:EmployeeId", e.ename AS "ipo:Name", e.job AS "ipo:Job" , e.mgr AS "ipo:Manager", e.hiredate AS "ipo:HireDate", e.sal AS "ipo:Salary", e.comm AS "ipo:Commission", XMLForest(d.deptno AS "ipo:DeptNo", d.dname AS "ipo:DeptName", d.loc AS "ipo:Location") AS "ipo:Dept")) FROM emp e, dept d WHERE e.deptno = d.deptno; -- creates the XML instance with the correct namespace, prefixes and target schema -- location, and can be used as the query in the view definition: -- <ipo:Employee xmlns="http://www.oracle.com/emp.xsd" -- xmlns:xsi="http://www.oracle.com/emp.xsd -- http://www.oracle.com/emp.xsd"> -- <ipo:EmployeeId>2100</ipo:EmployeeId> -- <ipo:Name>John</ipo:Name> -- <ipo:Manager>Mary</ipo:Manager> -- <ipo:Hiredate>12-Jan-01</ipo:Hiredate> -- <ipo:Salary>123003</ipo:Salary> -- <ipo:Dept> -- <ipo:Deptno>2000</ipo:Deptno> -- <ipo:DeptName>Sports</ipo:DeptName> -- <ipo:Location>San Francisco</ipo:Location> -- </ipo:Dept> -- </ipo:Employee>
If the XML schema had no target namespace then you can use the xsi:noNamespaceSchemaLocation attribute to denote that. For example, consider the following XML schema that is registered at location: "emp-noname.xsd":
BEGIN dbms_xmlschema.deleteSchema('emp-noname.xsd', 4); END; / BEGIN dbms_xmlschema.registerSchema('emp-noname.xsd', '<xs:schema xmlns:xs="http://www.w3.org/2001/XMLSchema" xmlns:xdb="http://xmlns.oracle.com/xdb"> <xs:element name = "Employee" xdb:defaultTable="EMP37_TAB"> <xs:complexType> <xs:sequence> <xs:element name = "EmployeeId" type = "xs:positiveInteger"/> <xs:element name = "FirstName" type = "xs:string"/> <xs:element name = "LastName" type = "xs:string"/> <xs:element name = "Salary" type = "xs:positiveInteger"/> </xs:sequence> </xs:complexType> </xs:element> </xs:schema>'); END; /
The following CREATE OR REPLACE VIEW statement creates a view that conforms to this XML schema:
CREATE OR REPLACE VIEW emp_xml OF XMLTYPE XMLSCHEMA "emp-noname.xsd" ELEMENT "Employee" WITH OBJECT ID (extract(sys_nc_rowinfo$, '/Employee/EmployeeId/text()').getnumberval()) AS SELECT XMLElement("Employee", XMLAttributes('http://www.w3.org/2001/XMLSchema-instance' AS "xmlns:xsi", 'emp-noname.xsd' AS "xsi:noNamespaceSchemaLocation"), XMLForest(e.empno AS "EmployeeId", e.ename AS "Name", e.job AS "Job" , e.mgr AS "Manager", e.hiredate AS "HireDate", e.sal AS "Salary", e.comm AS "Commission", XMLForest(d.deptno AS "DeptNo", d.dname AS "DeptName", d.loc AS "Location") AS "Dept")) FROM emp e, dept d WHERE e.deptno = d.deptno;
The XMLAttributes clause creates an XML element which contains the noNamespace schema location attribute.
For the second example view, to wrap the relational department data with nested employee information as XML, follow these steps:
DROP TYPE emp_t FORCE; DROP TYPE dept_t FORCE; CREATE OR REPLACE TYPE emp_t AS OBJECT ( EMPNO NUMBER(4), ENAME VARCHAR2(10), JOB VARCHAR2(9), MGR NUMBER(4), HIREDATE DATE, SAL NUMBER(7,2), COMM NUMBER(7,2) ); / CREATE OR REPLACE TYPE emplist_t AS TABLE OF emp_t; / CREATE OR REPLACE TYPE dept_t AS OBJECT ( DEPTNO NUMBER(2), DNAME VARCHAR2(14), LOC VARCHAR2(13), EMPS EMPLIST_T ); /
You can either use a pre-existing XML schema or you can generate an XML schema from the object type using the DBMS_XMLSchema.generateSchema(s) functions:
BEGIN dbms_xmlschema.deleteSchema('http://www.oracle.com/dept.xsd', 4); END; / BEGIN dbms_xmlschema.registerSchema('http://www.oracle.com/dept.xsd', '<schema xmlns="http://www.w3.org/2001/XMLSchema" targetNamespace="http://www.oracle.com/dept.xsd" version="1.0" xmlns:xdb="http://xmlns.oracle.com/xdb" elementFormDefault="qualified"> <element name = "Department" xdb:SQLType="DEPT_T" xdb:SQLSchema="SCOTT"> <complexType> <sequence> <element name = "DeptNo" type = "positiveInteger" xdb:SQLName="DEPTNO" xdb:SQLType="NUMBER"/> <element name = "DeptName" type = "string" xdb:SQLName="DNAME" xdb:SQLType="VARCHAR2"/> <element name = "Location" type = "string" xdb:SQLName="LOC" xdb:SQLType="VARCHAR2"/> <element name = "Employee" maxOccurs = "unbounded" xdb:SQLName = "EMPS" xdb:SQLType="EMPLIST_T" xdb:SQLSchema="SCOTT"> <complexType> <sequence> <element name = "EmployeeId" type = "positiveInteger" xdb:SQLName="EMPNO" xdb:SQLType="NUMBER"/> <element name = "Name" type = "string" xdb:SQLName="ENAME" xdb:SQLType="VARCHAR2"/> <element name = "Job" type = "string" xdb:SQLName="JOB" xdb:SQLType="VARCHAR2"/> <element name = "Manager" type = "positiveInteger" xdb:SQLName="MGR" xdb:SQLType="NUMBER"/> <element name = "HireDate" type = "date" xdb:SQLName="HIREDATE" xdb:SQLType="DATE"/> <element name = "Salary" type = "positiveInteger" xdb:SQLName="SAL" xdb:SQLType="NUMBER"/> <element name = "Commission" type = "positiveInteger" xdb:SQLName="COMM" xdb:SQLType="NUMBER"/> </sequence> </complexType> </element> </sequence> </complexType> </element> </schema>', TRUE, FALSE, FALSE); END; /
Create the dept_xml XMLType view from the department object type as follows:
CREATE OR REPLACE VIEW dept_xml OF XMLTYPE XMLSCHEMA "http://www.oracle.com/dept.xsd" ELEMENT "Department" WITH OBJECT ID (EXTRACTVALUE(sys_nc_rowinfo$, '/Department/DeptNo')) AS SELECT dept_t(d.deptno, d.dname, d.loc, cast(multiset( SELECT emp_t(e.empno, e.ename, e.job, e.mgr, e.hiredate, e.sal,e.comm) FROM emp e WHERE e.deptno = d.deptno) AS emplist_t)) FROM dept d;
Create the dept_xml XMLType view from the relational tables without object types:
CREATE OR REPLACE VIEW dept_xml OF XMLTYPE XMLSCHEMA "http://www.oracle.com/dept.xsd" ELEMENT "Department" WITH OBJECT ID (EXTRACT(sys_nc_rowinfo$, '/Department/DeptNo').getNumberVal()) AS SELECT XMLElement("Department", XMLAttributes( 'http://www.oracle.com/emp.xsd' AS "xmlns" , 'http://www.w3.org/2001/XMLSchema-instance' AS "xmlns:xsi", 'http://www.oracle.com/dept.xsd http://www.oracle.com/dept.xsd' AS "xsi:schemaLocation"), XMLForest(deptno "DeptNo", d.dname "DeptName", d.loc "Location"), (SELECT XMLAGG(XMLElement("Employee", XMLForest(e.empno "EmployeeId", e.ename "Name", e.job "Job", e.mgr "Manager", e.hiredate "Hiredate"), e.sal "Salary",e.comm "Commission")) FROM emp e WHERE e.deptno = d.deptno)) FROM dept d; -- Step 3c. Create XMLType Views on XMLType Tables -- An XMLType view can be created on an XMLType table, perhaps to transform the XML -- or restrict the rows returned by some predicates, as follows: DROP TABLE dept_xml_tab; CREATE TABLE dept_xml_tab OF XMLTYPE XMLSCHEMA "http://www.oracle.com/dept.xsd" ELEMENT "Department" nested table xmldata."EMPS" store as dept_xml_tab_tab1; CREATE OR REPLACE VIEW dept_xml_view OF XMLTYPE XMLSCHEMA "http://www.oracle.com/dept.xsd" ELEMENT "Department" AS SELECT VALUE(p) FROM dept_xml_tab p WHERE Extractvalue(value(p), '/DeptNo') = 10;
| Note: The XML schema and element information must be specified at the view level because the SELECT list could arbitrarily construct XML of a different XML schema from the underlying table. | 
You can also create XMLType views by transforming XMLType tables.
For example, consider the creation of XMLType table po_tab. Refer to Example 6-1, "Transforming an XMLType Instance Using XMLTransform() and DBUriType to Get the XSL Stylesheet" for an xmltransform() example:
DROP TABLE po_tab; CREATE TABLE po_tab OF xmltype xmlschema "ipo.xsd" element "PurchaseOrder"; -- You can then create a view of the table as follows: CREATE OR REPLACE VIEW HR_PO_tab OF xmltype xmlschema "hrpo.xsd" element "PurchaseOrder" WITH OBJECT ID DEFAULT AS SELECT xmltransform(value(p),xdburitype('/home/SCOTT/xsl/po2.xsl').getxml()) FROM po_tab p;
You can reference an XMLType view object using the REF() syntax:
SELECT REF(p) FROM dept_xml p;
XMLType view reference REF() is based on one of the following object IDs:
XMLType tables or object viewsThese REFs can be used to fetch OCIXMLType instances in the OCI Object cache or can be used inside SQL queries. These REFs behave in the same way as REFs to object views.
An XMLType view may not be inherently updatable. This means that you have to write INSTEAD-OF -TRIGGERS to handle all data manipulation (DML). You can identify cases where the view is implicitly updatable, by analyzing the underlying view query.
For example, if the XMLType view query is based on an object view or an object constructor that is itself inherently updatable:
DROP TYPE dept_t force; CREATE OR REPLACE TYPE dept_t AS OBJECT ( DEPTNO NUMBER(2), DNAME VARCHAR2(14), LOC VARCHAR2(13) ); / BEGIN dbms_xmlschema.deleteSchema('http://www.oracle.com/dept.xsd', 4); END; / BEGIN dbms_xmlschema.registerSchema('http://www.oracle.com/dept.xsd', '<schema xmlns="http://www.w3.org/2001/XMLSchema" targetNamespace="http://www.oracle.com/dept.xsd" version="1.0" xmlns:xdb="http://xmlns.oracle.com/xdb" elementFormDefault="qualified"> <element name = "Department" xdb:SQLType="DEPT_T" xdb:SQLSchema="SCOTT"> <complexType> <sequence> <element name = "DeptNo" type = "positiveInteger" xdb:SQLName="DEPTNO" xdb:SQLType="NUMBER"/> <element name = "DeptName" type = "string" xdb:SQLName="DNAME" xdb:SQLType="VARCHAR2"/> <element name = "Location" type = "string" xdb:SQLName="LOC" xdb:SQLType="VARCHAR2"/> </sequence> </complexType> </element> </schema>', TRUE, FALSE, FALSE); END; / CREATE OR REPLACE VIEW dept_xml of xmltype xmlschema "http://www.oracle.com/dept.xsd" element "Department" with object id (sys_nc_rowinfo$.extract('/Department/DeptNo').getnumberval()) as select dept_t(d.deptno, d.dname, d.loc) from dept d; INSERT INTO dept_xml VALUES (XMLType.createXML( '<Department xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:noNamespaceSchemaLocation="http://www.oracle.com/dept.xsd"> <DeptNo>50</DeptNo> <DeptName>EDP</DeptName> <Location>NEW YORK</Location> </Department>')); UPDATE dept_xml d SET d.sys_nc_rowinfo$ = updateXML(d.sys_nc_rowinfo$, '/Department/DeptNo/text()', 60) WHERE existsNode(d.sys_nc_rowinfo$, '/Department[DeptNo=50]') = 1;
For Query Rewrites, XMLType views are the same as regular XMLType table columns. Hence, extract() or existsNode() operations on view columns, get rewritten into underlying relational accesses for better performance.
For example consider the following:
XCREATE OR REPLACE VIEW dept_ov OF dept_t WITH OBJECT ID (deptno) as SELECT d.deptno, d.dname, d.loc, cast(multiset( SELECT emp_t(e.empno, e.ename, e.job, e.mgr, e.hiredate, e.sal, e.comm) FROM emp e WHERE e.deptno = d.deptno) AS emplist_t) FROM dept d; CREATE OR REPLACE VIEW dept_xml OF XMLTYPE WITH OBJECT ID (EXTRACT(sys_nc_rowinfo$, '/ROW/DEPTNO').getNumberVal()) AS SELECT sys_xmlgen(value(p)) FROM dept_ov p;
A query to select department numbers that have at least one employee making a salary more than $200000:
SELECT EXTRACTVALUE(value(x),'/ROW/DEPTNO') FROM dept_xml x WHERE EXISTSNODE(value(x), '/ROW/EMPS/EMP_T[SAL > 200]') = 1;
becomes:
ELECT d.deptno FROM dept d WHERE EXISTS (SELECT NULL FROM emp e WHERE e.deptno = d.deptno AND e.sal > 200);
Consider the following example:
Non-schema-based XMLType views can be created on existing relational and object-relational tables and views. This provides users an XML view of the underlying data.
Existing relational data can be transformed into XMLType views by creating appropriate types, and doing a SYS_XMLGEN at the top-level.For example, the data in the emp table can be exposed as follows :
CREATE TYPE Emp_t AS OBJECT ( EMPNO NUMBER(4), ENAME VARCHAR2(10), JOB VARCHAR2(9), MGR NUMBER(4), HIREDATE DATE, SAL NUMBER(7,2), COMM NUMBER(7,2)); CREATE VIEW employee_xml OF XMLTYPE WITH OBJECT ID (SYS_NC_ROWINFO$.EXTRACT('/ROW/EMPNO/text()').getnumberval()) AS SELECT SYS_XMLGEN( emp_t(e.empno, e.ename, e.job, e.mgr, e.hiredate, e.sal, e.comm)) FROM emp e;
A major advantage of non-schema-based views is that existing object views can be easily transformed into XMLType views without any additional DDLs. For example, consider a database which contains the object view employee_ov with the following defintion :
CREATE VIEW employee_ov OF EMP_T WITH OBJECT ID (empno) AS SELECT emp_t(e.empno, e.ename, e.job, e.mgr, e.hiredate, e.sal, e.comm) FROM emp e; -- Creating a non-schema-based XMLType views can be achieved by simply -- calling SYS_XMLGEN over the top-level object column. No additional -- types need to be created. CREATE OR REPLACE VIEW employee_ov_xml OF XMLTYPE WITH OBJECT ID (SYS_NC_ROWINFO$.EXTRACT('/ROW/EMPNO/text()').getnumberval()) AS SELECT SYS_XMLGEN(value(x)) from employee_ov x; -- Certain kinds of queries on SYS_XMLGEN views are rewritten to
access the object attributes directly. Simple XPath traversals with existsNode(), extractValue(), and extract() are candidates for rewrite. See Chapter 5, "Structured Mapping of XMLType", "Query Rewrite with XML Schema-Based Structured Storage", for details on query rewrite.
| Note: Query rewrite only happens with  | 
For example, a query such as the following :
SELECT EXTRACT(VALUE(x), '/ROW/EMPNO') FROM employee_ov_xml x WHERE EXTRACTVALUE(value(x), '/ROW/ENAME') = 'SMITH';
is rewritten to :
SELECT SYS_XMLGEN(empno) FROM emp e WHERE e.ename = 'SMITH';
In the preceding examples, the CREATE VIEW statement specified the XML schema URL and element name, whereas the underlying view query simply constructed a non-XML schema-based XMLType. However, there are several scenarios where you may want to avoid the view creation step, but still need to construct XML schema-based XML.
To achieve this, you can use the following XML generation functions to optionally accept an XML schema URL and element name:
If the XML schema information is specified, the resulting XML is created to be XML schema-based:
SELECT XMLTYPE.createXML(dept_t(d.deptno, d.dname, d.loc, CAST(MULTISET(SELECT emp_t(e.empno, e.ename, e.job, e.mgr, e.hiredate, e.sal, e.comm) FROM emp e WHERE e.deptno = d.deptno) AS emplist_t), 'http://www.oracle.com/dept.xsd', 'Department') FROM dept d;
You can fill in the optional Oracle XML DB attributes before registering the XML schema. In this case, Oracle validates the extra information to ensure that the specified values for the Oracle XML DB attributes are compatible with the rest of the XML schema declarations. This form of XML schema registration typically happens when wrapping existing data using XMLType views.
| See: Chapter 5, "Structured Mapping of XMLType" for more details on this process | 
You can use the DBMS_XMLSchema generateSchema() and generateSchemas() functions to generate the default XML mapping for specified object types. The generated XML schema document has the SQLType, SQLSchema, and so on, attributes filled in. When these XML schema documents are then registered, the following validation forms can occur:
XMLType. For example, an XML string datatype can only be mapped to VARCHAR2s or Large Objects (LOBs).complexType, that is, the object type has the right number of attributes with the right datatypes.| 
 |  Copyright © 2002 Oracle Corporation. All Rights Reserved. | 
 |