Especificar la profundidad en relaciones recursivas utilizando sql:max-depth
Se aplica a: SQL Server Azure SQL Database
En las bases de datos relacionales, cuando una tabla se relaciona consigo misma, este tipo de relación recibe el nombre de relación recursiva. Por ejemplo, en una relación supervisor-supervisado, una tabla que almacena los registros de empleados se relaciona consigo misma. En este caso, la tabla de empleados desempeña un rol de supervisor en uno de los lados de la relación y un rol de supervisado en el otro lado.
Los esquemas de asignación pueden incluir relaciones recursivas donde un elemento y su antecesor son del mismo tipo.
Ejemplo A
Fíjese en la siguiente tabla:
Emp (EmployeeID, FirstName, LastName, ReportsTo)
En esta tabla, la columna ReportsTo almacena el identificador de empleado del director.
Supongamos que desea crear una jerarquía XML de empleados en la que el empleado director se sitúa en la parte superior de la jerarquía y los empleados que son subordinados directos de ese director aparecen en la jerarquía correspondiente tal y como se muestra en el siguiente fragmento XML de ejemplo. Lo que muestra este fragmento es el árbol recursivo para el empleado 1.
<?xml version="1.0" encoding="utf-8" ?>
<root>
<Emp FirstName="Nancy" EmployeeID="1" LastName="Devolio">
<Emp FirstName="Andrew" EmployeeID="2" LastName="Fuller" />
<Emp FirstName="Janet" EmployeeID="3" LastName="Leverling">
<Emp FirstName="Margaret" EmployeeID="4" LastName="Peacock">
<Emp FirstName="Steven" EmployeeID="5" LastName="Devolio">
...
...
</root>
En este fragmento, el empleado 5 es subordinado directo del empleado 4, el empleado 4 es subordinado directo del empleado 3 y los empleados 3 y 2 son subordinados directos del empleado 1.
Para generar este resultado, puede usar el siguiente esquema XSD y especificar una consulta XPath en él. El esquema describe un <elemento Emp> de tipo EmployeeType, que consta de un <elemento secundario Emp> del mismo tipo, EmployeeType. Se trata de una relación recursiva (el elemento y su antecesor son del mismo tipo). Además, el esquema usa una <relación sql:relationship> para describir la relación de elementos primarios y secundarios entre el supervisor y el monitor. Tenga en cuenta que en esta <sql:relationship>, Emp es la tabla primaria y secundaria.
<xsd:schema xmlns:xsd="http://www.w3.org/2001/XMLSchema"
xmlns:dt="urn:schemas-microsoft-com:datatypes"
xmlns:sql="urn:schemas-microsoft-com:mapping-schema">
<xsd:annotation>
<xsd:appinfo>
<sql:relationship name="SupervisorSupervisee"
parent="Emp"
parent-key="EmployeeID"
child="Emp"
child-key="ReportsTo" />
</xsd:appinfo>
</xsd:annotation>
<xsd:element name="Emp" type="EmployeeType"
sql:relation="Emp"
sql:key-fields="EmployeeID"
sql:limit-field="ReportsTo" />
<xsd:complexType name="EmployeeType">
<xsd:sequence>
<xsd:element name="Emp" type="EmployeeType"
sql:relation="Emp"
sql:key-fields="EmployeeID"
sql:relationship="SupervisorSupervisee"
sql:max-depth="6" />
</xsd:sequence>
<xsd:attribute name="EmployeeID" type="xsd:ID" />
<xsd:attribute name="FirstName" type="xsd:string"/>
<xsd:attribute name="LastName" type="xsd:string"/>
</xsd:complexType>
</xsd:schema>
Dado que la relación es recursiva, necesita algún modo de especificar la profundidad de recursión del esquema. De lo contrario, el resultado será una recursión sin fin (empleado subordinado directo de empleado subordinado directo de empleado, etc.). La anotación sql:max-depth permite especificar la profundidad en la recursividad que se va a realizar. En este ejemplo concreto, para especificar un valor para sql:max-depth, debe saber qué profundidad pasa la jerarquía de administración en la empresa.
Nota:
El esquema especifica la anotación sql:limit-field , pero no especifica la anotación sql:limit-value . Esto limita el nodo superior de la jerarquía resultante únicamente a los empleados que no son subordinados directos de nadie. (ReportsTo es NULL). Especificar sql:limit-field y no especificar la anotación sql:limit-value (que el valor predeterminado es NULL) logra esto. Si desea que el XML resultante incluya todos los árboles de informes posibles (el árbol de informes para cada empleado de la tabla), quite la anotación sql:limit-field del esquema.
Nota:
El siguiente procedimiento usa la base de datos tempdb.
Para probar una consulta XPath de ejemplo en el esquema
Cree una tabla de ejemplo denominada Emp en la base de datos tempdb a la que apunta la raíz virtual.
USE tempdb CREATE TABLE Emp ( EmployeeID int primary key, FirstName varchar(20), LastName varchar(20), ReportsTo int)
Agregue estos datos de ejemplo:
INSERT INTO Emp values (1, 'Nancy', 'Devolio',NULL) INSERT INTO Emp values (2, 'Andrew', 'Fuller',1) INSERT INTO Emp values (3, 'Janet', 'Leverling',1) INSERT INTO Emp values (4, 'Margaret', 'Peacock',3) INSERT INTO Emp values (5, 'Steven', 'Devolio',4) INSERT INTO Emp values (6, 'Nancy', 'Buchanan',5) INSERT INTO Emp values (7, 'Michael', 'Suyama',6)
Copie el código de esquema anterior y péguelo en un archivo de texto. Guarde el archivo como maxDepth.xml.
Copie la plantilla siguiente y péguela en un archivo de texto. Guarde el archivo como maxDepthT.xml en el mismo directorio donde guardó maxDepth.xml. La consulta de la plantilla devuelve todos los empleados de la tabla Emp.
<ROOT xmlns:sql="urn:schemas-microsoft-com:xml-sql"> <sql:xpath-query mapping-schema="maxDepth.xml"> /Emp </sql:xpath-query> </ROOT>
La ruta de acceso al directorio especificada para el esquema de asignación (maxDepth.xml) es relativa al directorio donde se guarda la plantilla. También puede especificarse una ruta de acceso absoluta como, por ejemplo:
mapping-schema="C:\MyDir\maxDepth.xml"
Cree y use el script de prueba SQLXML 4.0 (Sqlxml4test.vbs) para ejecutar la plantilla. Para obtener más información, consulte Uso de ADO para ejecutar consultas SQLXML 4.0.
El resultado es el siguiente:
<?xml version="1.0" encoding="utf-8" ?>
<root>
<Emp FirstName="Nancy" EmployeeID="1" LastName="Devolio">
<Emp FirstName="Andrew" EmployeeID="2" LastName="Fuller" />
<Emp FirstName="Janet" EmployeeID="3" LastName="Leverling">
<Emp FirstName="Margaret" EmployeeID="4" LastName="Peacock">
<Emp FirstName="Steven" EmployeeID="5" LastName="Devolio">
<Emp FirstName="Nancy" EmployeeID="6" LastName="Buchanan">
<Emp FirstName="Michael" EmployeeID="7" LastName="Suyama" />
</Emp>
</Emp>
</Emp>
</Emp>
</Emp>
</root>
Nota:
Para generar diferentes profundidades de jerarquías en el resultado, cambie el valor de la anotación sql:max-depth en el esquema y vuelva a ejecutar la plantilla después de cada cambio.
En el esquema anterior, todos los <elementos Emp> tenían exactamente el mismo conjunto de atributos (EmployeeID, FirstName y LastName). El esquema siguiente se ha modificado ligeramente para devolver un atributo ReportsTo adicional para todos los <elementos Emp> que informan a un administrador.
Por ejemplo, este fragmento XML muestra los subordinados del empleado 1:
<?xml version="1.0" encoding="utf-8" ?>
<root>
<Emp FirstName="Nancy" EmployeeID="1" LastName="Devolio">
<Emp FirstName="Andrew" EmployeeID="2"
ReportsTo="1" LastName="Fuller" />
<Emp FirstName="Janet" EmployeeID="3"
ReportsTo="1" LastName="Leverling">
...
...
Éste es el esquema revisado:
<xsd:schema xmlns:xsd="http://www.w3.org/2001/XMLSchema"
xmlns:dt="urn:schemas-microsoft-com:datatypes"
xmlns:sql="urn:schemas-microsoft-com:mapping-schema">
<xsd:annotation>
<xsd:documentation>
Customer-Order-Order Details Schema
Copyright 2000 Microsoft. All rights reserved.
</xsd:documentation>
<xsd:appinfo>
<sql:relationship name="SupervisorSupervisee"
parent="Emp"
parent-key="EmployeeID"
child="Emp"
child-key="ReportsTo" />
</xsd:appinfo>
</xsd:annotation>
<xsd:element name="Emp"
type="EmpType"
sql:relation="Emp"
sql:key-fields="EmployeeID"
sql:limit-field="ReportsTo" />
<xsd:complexType name="EmpType">
<xsd:sequence>
<xsd:element name="Emp"
type="EmpType"
sql:relation="Emp"
sql:key-fields="EmployeeID"
sql:relationship="SupervisorSupervisee"
sql:max-depth="6"/>
</xsd:sequence>
<xsd:attribute name="EmployeeID" type="xsd:int" />
<xsd:attribute name="FirstName" type="xsd:string"/>
<xsd:attribute name="LastName" type="xsd:string"/>
<xsd:attribute name="ReportsTo" type="xsd:int" />
</xsd:complexType>
</xsd:schema>
Anotación sql:max-depth
En un esquema compuesto de relaciones recursivas, la profundidad de recursión debe especificarse de forma explícita en el esquema. Esto es necesario para generar correctamente la consulta FOR XML EXPLICIT correspondiente que devuelve los resultados solicitados.
Use la anotación sql:max-depth en el esquema para especificar la profundidad de recursión en una relación recursiva que se describe en el esquema. El valor de la anotación sql:max-depth es un entero positivo (de 1 a 50) que indica el número de recursiones: un valor de 1 detiene la recursividad en el elemento para el que se especifica la anotación sql:max-depth ; un valor de 2 detiene la recursividad en el siguiente nivel del elemento en el que se especifica sql:max-depth ; etc.
Nota:
En la implementación subyacente, una consulta XPath que se especifica en un esquema de asignación se convierte en una consulta SELECT ... FOR XML EXPLICIT. Esta consulta exige que se especifique una profundidad finita de recursión. Cuanto mayor sea el valor especificado para sql:max-depth, mayor será la consulta FOR XML EXPLICIT que se genera. Esto puede ralentizar el tiempo de recuperación.
Nota:
Los diagramas de actualización y la carga masiva XML omiten la anotación max-depth. Esto significa que se llevarán a cabo inserciones o actualizaciones recursivas independientemente del valor que se especifique para max-depth.
Especificar sql:max-depth en elementos complejos
La anotación sql:max-depth se puede especificar en cualquier elemento de contenido complejo.
Elementos recursivos
Si se especifica sql:max-depth en el elemento primario y en el elemento secundario en una relación recursiva, la anotación sql:max-depth especificada en el elemento primario tiene prioridad. Por ejemplo, en el esquema siguiente, se especifica la anotación sql:max-depth en los elementos primarios y secundarios employee. En este caso, sql:max-depth=4, especificado en el <elemento primario Emp> (desempeñando un rol de supervisor), tiene prioridad. Se omite la profundidad sql:max especificada en el elemento Emp> secundario <(desempeñando un rol de supervisión).
Ejemplo B
<xsd:schema xmlns:xsd="http://www.w3.org/2001/XMLSchema"
xmlns:dt="urn:schemas-microsoft-com:datatypes"
xmlns:sql="urn:schemas-microsoft-com:mapping-schema">
<xsd:annotation>
<xsd:appinfo>
<sql:relationship name="SupervisorSupervisee"
parent="Emp"
parent-key="EmployeeID"
child="Emp"
child-key="ReportsTo" />
</xsd:appinfo>
</xsd:annotation>
<xsd:element name="Emp" type="EmployeeType"
sql:relation="Emp"
sql:key-fields="EmployeeID"
sql:limit-field="ReportsTo"
sql:max-depth="3" />
<xsd:complexType name="EmployeeType">
<xsd:sequence>
<xsd:element name="Emp" type="EmployeeType"
sql:relation="Emp"
sql:key-fields="EmployeeID"
sql:relationship="SupervisorSupervisee"
sql:max-depth="2" />
</xsd:sequence>
<xsd:attribute name="EmployeeID" type="xsd:ID" />
<xsd:attribute name="FirstName" type="xsd:string"/>
<xsd:attribute name="LastName" type="xsd:string"/>
</xsd:complexType>
</xsd:schema>
Para probar este esquema, siga los pasos proporcionados para el ejemplo A, anteriormente en este tema.
Elementos no recursivos
Si se especifica la anotación sql:max-depth en un elemento del esquema que no causa ninguna recursividad, se omite. En el esquema siguiente, un <elemento Emp> consta de un< elemento secundario Constant>, que, a su vez, tiene un <elemento secundario Emp>.
En este esquema, se omite la anotación sql:max-depth especificada en el< elemento Constant> porque no hay recursividad entre el <elemento primario Emp> y el< elemento secundario Constant.> Pero hay recursividad entre el antecesor Emp> y el <niño Emp>.< El esquema especifica la anotación sql:max-depth en ambos. Por lo tanto, la anotación sql:max-depth que se especifica en el antecesor (<Emp> en el rol de supervisor) tiene prioridad.
Ejemplo C
<xsd:schema xmlns:xsd="http://www.w3.org/2001/XMLSchema"
xmlns:sql="urn:schemas-microsoft-com:mapping-schema">
<xsd:annotation>
<xsd:appinfo>
<sql:relationship name="SupervisorSupervisee"
parent="Emp"
child="Emp"
parent-key="EmployeeID"
child-key="ReportsTo"/>
</xsd:appinfo>
</xsd:annotation>
<xsd:element name="Emp"
sql:relation="Emp"
type="EmpType"
sql:limit-field="ReportsTo"
sql:max-depth="1" />
<xsd:complexType name="EmpType" >
<xsd:sequence>
<xsd:element name="Constant"
sql:is-constant="1"
sql:max-depth="20" >
<xsd:complexType >
<xsd:sequence>
<xsd:element name="Emp"
sql:relation="Emp" type="EmpType"
sql:relationship="SupervisorSupervisee"
sql:max-depth="3" />
</xsd:sequence>
</xsd:complexType>
</xsd:element>
</xsd:sequence>
<xsd:attribute name="EmployeeID" type="xsd:int" />
</xsd:complexType>
</xsd:schema>
Para probar este esquema, siga los pasos que se proporcionaron para el Ejemplo A anteriormente en este tema.
Tipos complejos derivados por restricción
Si tiene una derivación de tipo complejo por <restricción>, los elementos del tipo complejo base correspondiente no pueden especificar la anotación sql:max-depth . En estos casos, la anotación sql:max-depth se puede agregar al elemento del tipo derivado.
Por otro lado, si tiene una derivación de tipo complejo por <extensión>, los elementos del tipo complejo base correspondiente pueden especificar la anotación sql:max-depth .
Por ejemplo, el siguiente esquema XSD genera un error porque se especifica la anotación sql:max-depth en el tipo base. Esta anotación no se admite en un tipo derivado de <la restricción> de otro tipo. Para corregir este problema, debe cambiar el esquema y especificar la anotación sql:max-depth en el elemento del tipo derivado.
Ejemplo D
<xsd:schema xmlns:xsd="http://www.w3.org/2001/XMLSchema"
xmlns:dt="urn:schemas-microsoft-com:datatypes"
xmlns:msdata="urn:schemas-microsoft-com:mapping-schema">
<xsd:complexType name="CustomerBaseType">
<xsd:sequence>
<xsd:element name="CID" msdata:field="CustomerID" />
<xsd:element name="CompanyName"/>
<xsd:element name="Customers" msdata:max-depth="3">
<xsd:annotation>
<xsd:appinfo>
<msdata:relationship
parent="Customers"
parent-key="CustomerID"
child-key="CustomerID"
child="Customers" />
</xsd:appinfo>
</xsd:annotation>
</xsd:element>
</xsd:sequence>
</xsd:complexType>
<xsd:element name="Customers" type="CustomerType"/>
<xsd:complexType name="CustomerType">
<xsd:complexContent>
<xsd:restriction base="CustomerBaseType">
<xsd:sequence>
<xsd:element name="CID"
type="xsd:string"/>
<xsd:element name="CompanyName"
type="xsd:string"
msdata:field="CName" />
<xsd:element name="Customers"
type="CustomerType" />
</xsd:sequence>
</xsd:restriction>
</xsd:complexContent>
</xsd:complexType>
</xsd:schema>
En el esquema, se especifica sql:max-depth en un tipo complejo CustomerBaseType . El esquema también especifica un <elemento Customer> de tipo CustomerType, que se deriva de CustomerBaseType. Una consulta XPath especificada en este esquema generará un error, ya que sql:max-depth no se admite en un elemento definido en un tipo base de restricción.
Esquemas con una jerarquía profunda
Puede tener un esquema que incluya una jerarquía profunda en la que un elemento contiene un elemento secundario, que a su vez contiene otro elemento secundario, y así sucesivamente. Si la anotación sql:max-depth especificada en este esquema genera un documento XML que incluye una jerarquía de más de 500 niveles (con el elemento de nivel superior en el nivel 1, su elemento secundario en el nivel 2, etc.), se devuelve un error.