XML元素和属性的多个名称

时间:2018-06-11 07:41:08

标签: c# xml xmlserializer

不是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查找我正在寻找的属性。

1 个答案:

答案 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; }

    }
}