首页 > 解决方案 > XML 元素和属性的多个名称

问题描述

NOT A DUPLICATE OF THIS这已被标记为我添加到现有 StackOverflow 问题和解决方案的链接的副本。这不是重复的,因为该问题专门针对 XmlElements。我正在寻找一种方法来塑造该解决方案以使用 XmlAttributes 以及 XmlElements。


我正在构建一个与返回 XML 的 Web 服务交互的类库。此 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 的对象,而不是为每个项目返回不同的类型。

我在这里找到了一个很好的解决方案处理元素的同义词。唯一的问题是它没有显示我如何在属性中添加同义词。

我试图创建另一个为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解决方案。它将所有重要信息放在一起,并使用反射来查找我正在寻找的属性。

标签: c#xmlxmlserializer

解决方案


使用 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; }

    }
}

推荐阅读