Моя текущая задача заключается в написании библиотеки классов для обработки файлов HL7 CDA.
Эти файлы HL7 CDA являются файлами XML с определенной XML-схемой, поэтому я использовал xsd.exe для генерации классов .NET для сериализации и десериализации XML.
Схема XML содержит различные типы, которые содержат атрибут mixed = "true" , указывающий, что узел XML этого типа может содержать обычный текст, смешанный с другими узлами XML.
Соответствующая часть XML-схемы для одного из этих типов выглядит следующим образом:
<xs:complexType name="StrucDoc.Paragraph" mixed="true">
<xs:sequence>
<xs:element name="caption" type="StrucDoc.Caption" minOccurs="0"/>
<xs:choice minOccurs="0" maxOccurs="unbounded">
<xs:element name="br" type="StrucDoc.Br"/>
<xs:element name="sub" type="StrucDoc.Sub"/>
<xs:element name="sup" type="StrucDoc.Sup"/>
<!-- ...other possible nodes... -->
</xs:choice>
</xs:sequence>
<xs:attribute name="ID" type="xs:ID"/>
<!-- ...other attributes... -->
</xs:complexType>
Сгенерированный код для этого типа выглядит следующим образом:
/// <remarks/>
[System.CodeDom.Compiler.GeneratedCodeAttribute("xsd", "2.0.50727.3038")]
[System.SerializableAttribute()]
[System.Diagnostics.DebuggerStepThroughAttribute()]
[System.ComponentModel.DesignerCategoryAttribute("code")]
[System.Xml.Serialization.XmlTypeAttribute(TypeName="StrucDoc.Paragraph", Namespace="urn:hl7-org:v3")]
public partial class StrucDocParagraph {
private StrucDocCaption captionField;
private object[] itemsField;
private string[] textField;
private string idField;
// ...fields for other attributes...
/// <remarks/>
public StrucDocCaption caption {
get {
return this.captionField;
}
set {
this.captionField = value;
}
}
/// <remarks/>
[System.Xml.Serialization.XmlElementAttribute("br", typeof(StrucDocBr))]
[System.Xml.Serialization.XmlElementAttribute("sub", typeof(StrucDocSub))]
[System.Xml.Serialization.XmlElementAttribute("sup", typeof(StrucDocSup))]
// ...other possible nodes...
public object[] Items {
get {
return this.itemsField;
}
set {
this.itemsField = value;
}
}
/// <remarks/>
[System.Xml.Serialization.XmlTextAttribute()]
public string[] Text {
get {
return this.textField;
}
set {
this.textField = value;
}
}
/// <remarks/>
[System.Xml.Serialization.XmlAttributeAttribute(DataType="ID")]
public string ID {
get {
return this.idField;
}
set {
this.idField = value;
}
}
// ...properties for other attributes...
}
Если я десериализую элемент XML, где узел абзаца выглядит следующим образом:
<paragraph>first line<br /><br />third line</paragraph>
Результат состоит в том, что элемент и текстовые массивы читаются так:
itemsField = new object[]
{
new StrucDocBr(),
new StrucDocBr(),
};
textField = new string[]
{
"first line",
"third line",
};
Из этого невозможно определить точный порядок текста и других узлов.
Если я сериализую это снова, результат будет выглядеть примерно так:
<paragraph>
<br />
<br />first linethird line
</paragraph>
Сериализатор по умолчанию просто сериализует сначала элементы, а затем текст.
Я пытался реализовать IXmlSerializable
в классе StrucDocParagraph, чтобы я мог контролировать десериализацию и сериализацию контента, но это довольно сложно, так как задействовано так много классов, и я пока не пришел к решению, потому что я не не знаю, окупятся ли усилия.
Есть ли какой-нибудь простой обходной путь к этой проблеме, или это вообще возможно, выполняя пользовательскую сериализацию через IXmlSerializable
?
Или я должен просто использовать XmlDocument
или XmlReader
/ XmlWriter
для обработки этих документов?