不是THIS的重复:这已被标记为我添加到现有StackOverflow问题和解决方案的链接的副本。这不是重复,因为该问题专门处理XmlElements。我正在寻找一种方法来模拟solution以使用XmlAttributes以及XmlElements。
我正在构建一个与返回XML的Web Service交互的类库。此Web服务附加到文档管理系统。我为系统中的每种实体构建了一个对象(文件夹,文档,用户等)。我目前有一个问题,有三个不同的操作返回指定目录中的文档列表。每个操作都以不同的格式返回文档元数据。
<d id="1104" name="Intro.pdf" cdate="2018-06-08 13:27:05" size="188481" />
<d id="1104" n="Intro.pdf" s="188481" />
<document DocumentID="1104" Name="Intro.pdf" CreationDate="2018-06-08 13:27:05" Size="188481" />
所有这些元素都是针对完全相同的文档。我想将其中的每一个反序列化为名为Document的同一对象,而不是为每个项返回不同的类型。
我找到了一个很好的解决方案here,用于处理元素的同义词。唯一的问题是它没有显示我如何在属性中添加同义词。
我试图创建另一个为UnknownAttribute
事件调用的方法,但我没有成功。该方法只是上一个链接中 SynonymHandler 方法的副本,但我稍微改了一下。
protected void AttributeSynonymHandler(object sender, XmlAttributeEventArgs e)
{
//this part works as it returns the correct property to me
var member = SynonymsAttribute.GetMember(e.ObjectBeingDeserialized, e.Attr.Name);
Type memberType;
if (member != null && member is FieldInfo)
memberType = ((FieldInfo)member).FieldType;
else if (member != null && member is PropertyInfo)
memberType = ((PropertyInfo)member).PropertyType;
else
return;
if (member != null)
{
//this is where the logic falls down, mainly because I don't have the original element anymore.
object value;
XmlSynonymDeserializer serializer = new XmlSynonymDeserializer(memberType, new XmlRootAttribute(e.Attr.Name));
using (System.IO.StringReader reader = new System.IO.StringReader(e.Attr.OuterXml))
value = serializer.Deserialize(reader);
if (member is FieldInfo)
((FieldInfo)member).SetValue(e.ObjectBeingDeserialized, value);
else if (member is PropertyInfo)
((PropertyInfo)member).SetValue(e.ObjectBeingDeserialized, value);
}
}
我可能错了,但我真的希望我能避免在每种情况下手动处理属性。任何帮助表示赞赏。
编辑:当涉及到其他类型(例如文件夹)时,我也有同样的问题,所以我正在寻找一个解决方案,而不是必须为每个对象创建一个单独的反序列化器至。这就是为什么我真的很喜欢SynonymAttribute : Attribute
解决方案。它将所有重要信息保存在一起,并使用Reflection查找我正在寻找的属性。
答案 0 :(得分:0)
使用xml linq,您可以使用以下代码:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Xml;
using System.Xml.Linq;
namespace ConsoleApplication48
{
class Program
{
const string FILENAME = @"c:\temp\test.xml";
static void Main(string[] args)
{
XDocument doc = XDocument.Load(FILENAME);
var groups = (from d in doc.Descendants("d")
join xdoc in doc.Descendants("document") on (string)d.Attribute("id") equals (string)xdoc.Attribute("DocumentID")
select new List<XElement> { d, xdoc })
.GroupBy(x => (string)x.FirstOrDefault().Attribute("id"))
.Select(x => x.SelectMany(y => y).ToList())
.ToList();
List<Document> documents = new List<Document>();
foreach (var group in groups)
{
Document newDoc = new Document();
documents.Add(newDoc);
foreach (XElement element in group)
{
foreach (XAttribute attribute in element.Attributes())
{
switch (attribute.Name.LocalName.ToUpper())
{
case "ID" :
newDoc.id = (string)attribute;
break;
case "DOCUMENTID":
newDoc.id = (string)attribute;
break;
case "NAME":
newDoc.name = (string)attribute;
break;
case "N":
newDoc.name = (string)attribute;
break;
case "CDATE":
newDoc.date = (DateTime)attribute;
break;
case "CREATIONDATE":
newDoc.date = (DateTime)attribute;
break;
case "SIZE":
newDoc.size = (long)attribute;
break;
case "S":
newDoc.size = (long)attribute;
break;
default :
break;
}
}
}
}
}
}
public class Document
{
public string id { get; set; }
public string name {get; set; }
public long size { get; set; }
public DateTime date { get; set; }
}
}