xmlns ser qué que proyecto prefijo nombres namespace está espacio definido debe xml xsd namespaces

ser - Espacios de nombres XML y atributos



qué es el xmlns (4)

Estoy tratando de entender cómo funcionan los espacios de nombres en XML. Cuando tengo un elemento como foo: bar, los atributos a menudo no tienen espacios de nombres en ellos. Pero a veces lo harán. ¿El atributo está en el espacio de nombre del elemento, incluso cuando se ha declarado el espacio de nombre predeterminado? Mirando sobre xsd para xhtml parece que los atributos son parte del esquema y deberían estar en el espacio de nombres para xhtml, pero nunca se presentan de esa manera ...


Ejemplos para ilustrar usando la notación Clark , donde el prefijo del espacio de nombres se reemplaza con la URL del espacio de nombres entre llaves:

<bar xmlns:foo="http://www.foo.com/" foo:baz="baz" qux="qux"/> <bar xmlns="http://www.foo.com/" xmlns:foo="http://www.foo.com/" foo:baz="baz" qux="qux"/> <foo:bar xmlns="http://www.foo.com/" xmlns:foo="http://www.foo.com/" foo:baz="baz" qux="qux"/>

es

<{}bar {http://www.foo.com/}baz="baz" {}qux="qux"/> <{http://www.foo.com/}bar {http://www.foo.com/}baz="baz" {}qux="qux"/> <{http://www.foo.com/}bar {http://www.foo.com/}baz="baz" {}qux="qux"/>


Hay algo relacionado con este tema de atributos / espacios de nombres que me llevó algo de tiempo entender hoy cuando estaba trabajando en un XSD. Voy a compartir esta experiencia con usted en caso de que alguien tenga los mismos problemas.

En el Documento de esquema en el que estaba trabajando había un par de atributos globales a los que se refieren algunos elementos. Para simplificar las cosas, supongamos que este XSD del que hablo fue sobre un cliente .

Llamemos a uno de estos atributos globales Id . Y el elemento raíz que lo usa Cliente

Mi declaración XSD se veía así:

<?xml version="1.0" encoding="utf-8"?> <xs:schema xmlns="http://schemas.mycompany.com/Customer/V1" targetNamespace="http://schemas.mycompany.com/Customer/V1" xmlns:xs="http://www.w3.org/2001/XMLSchema">

La declaración de mi atributo Id se veía así:

<xs:attribute name="Id" type="xs:positiveInteger"/>

Y mi elemento Cliente usó el atributo de esta manera:

<xs:element name="Customer"> <xs:complexType> <xs:attribute ref="Id" use="required"/> <!-- some elements here --> </xs:complexType> </xs:element>

Ahora, digamos que quería declarar un documento XML del cliente como este:

<?xml version="1.0" encoding="utf-8"?> <Customer Id="1" xmlns="http://schemas.mycompany.com/Customer/V1"> <!-- ... other elements here --> </Customer>

Descubrí que no puedo: cuando el atributo se declara globalmente, no está en el mismo espacio de nombres que el elemento que lo hace referencia.

Descubrí que la única solución con el XSD definido así era declarar el espacio de nombres dos veces: una vez sin un prefijo para convertirlo en el espacio de nombre predeterminado para los elementos, y una vez con un prefijo para usarlo con los atributos. Así que así es como se vería así:

<?xml version="1.0" encoding="utf-8"?> <Customer cus:Id="1" xmlns="http://schemas.mycompany.com/Customer/V1" xmlns:cus="http://schemas.mycompany.com/Customer/V1"> <!-- ... other elements here --> </Customer>

Esto es tan poco práctico que decidí deshacerme de todos los atributos globales y declararlos localmente. Lo que en el caso del ejemplo que di aquí se habría visto así:

<xs:element name="Customer"> <xs:complexType> <xs:attribute name="Id" type="xs:positiveInteger" use="required"/> <!-- some elements here --> </xs:complexType> </xs:element>

Me resulta difícil encontrar algunas referencias sobre lo que estoy hablando aquí en la red. Eventualmente encontré esta publicación en el Stylus XSD Forum donde un tipo llamado Steen Lehmann sugirió declarar el atributo localmente o declararlo dentro de un grupo de atributos

"para que la declaración de atributo en sí ya no sea global"

Esta última solución tiene un sabor "hacky", así que decidí seguir con la primera solución y declarar todos mis atributos localmente.


La mayoría de las veces, los atributos no estarán en ningún espacio de nombres. La especificación del espacio de nombres dice ( énfasis mío):

Una declaración de espacio de nombres predeterminada se aplica a todos los nombres de elementos no prefijados dentro de su alcance. Las declaraciones predeterminadas del espacio de nombre no se aplican directamente a los nombres de los atributos ; la interpretación de los atributos no prefijados está determinada por el elemento en el que aparecen.

Hay una razón por la que la mayoría de los vocabularios XML usan atributos sin nombre de espacio:
Cuando sus elementos tienen un espacio de nombre y esos elementos tienen atributos, entonces no puede haber confusión: los atributos pertenecen a su elemento, que pertenece a su espacio de nombres. Agregar un prefijo de espacio de nombres a los atributos simplemente haría todo más detallado.

Entonces, ¿por qué existen los atributos de espacio de nombres?
Debido a que algunos vocabularios hacen un trabajo útil con la mayoría de los atributos, y pueden hacerlo cuando se mezclan con otros vocabularios. El ejemplo más conocido es XLink .

Por último, W3C XML Schema tiene una manera muy fácil ( <schema attributeFormDefault="qualified"> ) de declarar sus atributos como si estuvieran en un espacio de nombres, lo que obliga a prefijarlos en sus documentos, incluso cuando usa un espacio de nombres predeterminado.


Lea en 6.1 Espacio de nombres y 6.2 Espacio de nombres predeterminado en w3c.

Básicamente:

El alcance de una declaración de espacio de nombres que declara un prefijo se extiende desde el comienzo de la etiqueta de inicio en la que aparece hasta el final de la etiqueta final correspondiente

Sin embargo, el texto aquí no parece explicar si significa que a es foo: a o el espacio de nombres predeterminado en el contexto. Supongo que no se refiere a foo: a, sino más bien al espacio de nombres predeterminado de los documentos a. Teniendo en cuenta esta cita al menos:

Dicha declaración de espacio de nombres se aplica a todos los nombres de elementos y atributos dentro de su alcance cuyo prefijo coincida con el especificado en la declaración.

Es decir. el espacio de nombres "foo:" solo se aplica a los elementos con el prefijo foo: