我有这段代码:
[XmlType( "Metadata" )]
[Serializable]
public class MetadataContainer : List<MetadataBase>
{
}
[XmlType( "Meta" )]
[XmlInclude( typeof( ReadonlyMetadata ) )]
[Serializable]
public abstract class MetadataBase
{
}
[XmlType( "Readonly" )]
[Serializable]
public class ReadonlyMetadata : MetadataBase
{
}
[TestFixture]
public class SerializationTests
{
[Test]
public void Can_deserialize_with_known_type()
{
const string text = @"<Metadata xmlns:xsd=""http://www.w3.org/2001/XMLSchema"" xmlns:xsi=""http://www.w3.org/2001/XMLSchema-instance"">
<Meta xsi:type=""Readonly"" />
</Metadata>";
var serializer = new XmlSerializer( typeof( MetadataContainer ) );
var metas = (MetadataContainer)serializer.Deserialize( XmlReader.Create( new StringReader( text ) ) );
Assert.That( metas.Count, Is.EqualTo( 1 ) );
Assert.That( metas.First(), Is.InstanceOf<ReadonlyMetadata>() );
}
[Test]
public void Can_deserialize_with_unknown_type()
{
const string text = @"<Metadata xmlns:xsd=""http://www.w3.org/2001/XMLSchema"" xmlns:xsi=""http://www.w3.org/2001/XMLSchema-instance"">
<Meta xsi:type=""Hello"" />
</Metadata>";
var serializer = new XmlSerializer( typeof( MetadataContainer ) );
var metas = (MetadataContainer)serializer.Deserialize( XmlReader.Create( new StringReader( text ) ) );
Assert.That( metas.Count, Is.EqualTo( 0 ) );
}
}
第一次测试有效,但是当我运行第二次测试时,我得到了这个错误:
System.InvalidOperationException:XML文档中存在错误(2,9)。 ----&GT; System.InvalidOperationException:无法识别指定的类型:name ='Hello',namespace ='',at。
我希望忽略不识别的类型而不是出现此错误。有没有办法做到这一点?
答案 0 :(得分:3)
类似问题的通用解决方案:
看看unknown element event (link)和unknown attribute event (link),看看他们是否解决了问题,或者我们必须变脏。请继续阅读...
此问题的解决方案
请记住,我不知道你的任务是什么,AFAIK它正在将xml序列化到你的数据结构中。如果您可以更改数据结构,我建议您查看Linq2XML并为您的目的创建一个智能工厂。
[TestMethod]
public void TestLinq2Xml()
{
const string text = @"<Metadata xmlns:xsd=""http://www.w3.org/2001/XMLSchema"" xmlns:xsi=""http://www.w3.org/2001/XMLSchema-instance"">
<Meta xsi:type=""Readonly"" />
<Meta xsi:type=""Garbage"" />
</Metadata>";
// Get the "names" of all implementors of MetadataBase
var types = AppDomain.CurrentDomain.GetAssemblies().ToList()
.SelectMany(s => s.GetTypes())
.Where(p => typeof(MetadataBase).IsAssignableFrom(p) && !p.IsAbstract && !p.IsInterface)
.Where(t => t.GetCustomAttributes(typeof(XmlTypeAttribute), false).Any())
.Select(t => t.GetCustomAttributes(typeof(XmlTypeAttribute), false)
.Cast<XmlTypeAttribute>().First().TypeName);
// Create a parser
var parser = new XmlSerializer(typeof(MetadataBase));
// Create metadatacontainer to fill
var metas = new MetadataContainer();
// Fill it with matching from from the XML
metas.AddRange((from t in XDocument.Parse(text).Descendants("Meta")
where types.Contains(t.Attribute(XName.Get("type", "http://www.w3.org/2001/XMLSchema-instance")).Value)
select (MetadataBase)parser.Deserialize(t.CreateReader())).ToList());
// Should be one guy present
Assert.AreEqual(metas.Count, 1);
}
答案 1 :(得分:0)
捕获数组中的所有未知元素。您仍然可以使用它们并稍后尝试反序列化,但这样可以完成反序列化。在您定义的每个类中都需要这个,以便在您怀疑存在未知元素的位置反序列化元素。
每http://msdn.microsoft.com/en-us/library/system.xml.serialization.xmlanyelementattribute.aspx:
Public Class XClass
' Apply the XmlAnyElementAttribute to a field returning an array
' of XmlElement objects.
<XmlAnyElement()> Public AllElements() As XmlElement
End Class 'XClass