Czy mogę wymusić kolejność atrybutów XML za pomocą schematu?

Nasza aplikacja C ++ odczytuje dane konfiguracyjne z plików XML, które wyglądają mniej więcej tak:

<data>
 <value id="FOO1" name="foo1" size="10" description="the foo" ... />
 <value id="FOO2" name="foo2" size="10" description="the other foo" ... />
 ...
 <value id="FOO300" name="foo300" size="10" description="the last foo" ... />
</data>

Kompletna konfiguracja aplikacji składa się z ~ 2500 tych plików XML (co przekłada się na ponad 1,5 miliona par atrybutów klucz / wartość). Pliki XML pochodzą z wielu różnych źródeł / zespołów i są sprawdzane pod kątem schematu. Czasami jednak<value/> węzły wyglądają tak:

<value name="bar1" id="BAR1" description="the bar" size="20" ... />

albo to:

<value id="BAT1" description="the bat" name="bat1"  size="25" ... />

Aby przyspieszyć ten proces, używamyExpat analizować dokumenty XML. Expat udostępnia atrybuty jako tablicę - tak:

void ExpatParser::StartElement(const XML_Char* name, const XML_Char** atts)
{
 // The attributes are stored in an array of XML_Char* where:
 //  the nth element is the 'key'
 //  the n+1 element is the value
 //  the final element is NULL
 for (int i = 0; atts[i]; i += 2) 
 {
  std::string key = atts[i];
  std::string value = atts[i + 1];
  ProcessAttribute (key, value);
 }
}

To nakłada na nas całą odpowiedzialnośćProcessAttribute() funkcja do odczytu „klucza” i zdecydowania, co zrobić z wartością.Profilowanie aplikacji wykazało, że ~ 40% całkowitego czasu analizowania XML zajmuje się tymi atrybutami według nazwy / łańcucha.

Ogólny proces można znacznie przyspieszyć, gdybym mógł zagwarantować / wymusić kolejność atrybutów (na początek, brak porównań łańcuchów wProcessAttribute()). Na przykład, jeśli atrybut „id” byłzawsze pierwszy atrybut, z którym możemy poradzić sobie bezpośrednio:

void ExpatParser::StartElement(const XML_Char* name, const XML_Char** atts)
{
 // The attributes are stored in an array of XML_Char* where:
 //  the nth element is the 'key'
 //  the n+1 element is the value
 //  the final element is NULL
 ProcessID (atts[1]);
 ProcessName (atts[3]);
 //etc.
}

Zgodnie ze specyfikacjami schematu W3C mogę użyć<xs:sequence> w schemacie XML, aby wymusić kolejność elementów - ale wydaje się, że nie działa dla atrybutów - a może używam go niepoprawnie:

<xs:element name="data">
 <xs:complexType>
  <xs:sequence>
   <xs:element name="value" type="value_type" minOccurs="1" maxOccurs="unbounded" />
  </xs:sequence>
 </xs:complexType>
</xs:element>

<xs:complexType name="value_type">
 <!-- This doesn't work -->
 <xs:sequence>
  <xs:attribute name="id" type="xs:string" />
  <xs:attribute name="name" type="xs:string" />
  <xs:attribute name="description" type="xs:string" />
 </xs:sequence>
</xs:complexType>

Czy istnieje sposób wymuszenia kolejności atrybutów w dokumencie XML? Jeśli odpowiedź brzmi „nie” - czy ktoś może zasugerować alternatywę, która nie miałaby wielkiej kary za wydajność?

questionAnswers(8)

yourAnswerToTheQuestion