使用动态将 XML 反序列化为对象

Posted

技术标签:

【中文标题】使用动态将 XML 反序列化为对象【英文标题】:Deserialize XML To Object using Dynamic 【发布时间】:2012-11-22 04:30:43 【问题描述】:

是否可以将未知的 XML 反序列化为如下对象?

 var xml = @"<Students><Student><Name>Arul</Name><Mark>90</Mark></Student></Students>";

 var serializer = new XmlSerializer(typeof(DynamicObject));

 dynamic students = serializer.Deserialize(new XmlTextReader(new StringReader(xml)));

【问题讨论】:

你为什么不试试呢? 我试过它抛出异常,但同样的逻辑适用于 Json 为什么它不支持 xml Converting XML to a dynamic C# object的可能重复 【参考方案1】:

你可能想试试这个。

string xml = @"<Students>
                <Student ID=""100"">
                    <Name>Arul</Name>
                    <Mark>90</Mark>
                </Student>
                <Student>
                    <Name>Arul2</Name>
                    <Mark>80</Mark>
                </Student>
            </Students>";

dynamic students = DynamicXml.Parse(xml);

var id = students.Student[0].ID;
var name1 = students.Student[1].Name;

foreach(var std in students.Student)

    Console.WriteLine(std.Mark);


public class DynamicXml : DynamicObject

    XElement _root;
    private DynamicXml(XElement root)
    
        _root = root;
    

    public static DynamicXml Parse(string xmlString)
    
        return new DynamicXml(XDocument.Parse(xmlString).Root);
    

    public static DynamicXml Load(string filename)
    
        return new DynamicXml(XDocument.Load(filename).Root);
    

    public override bool TryGetMember(GetMemberBinder binder, out object result)
    
        result = null;

        var att = _root.Attribute(binder.Name);
        if (att != null)
        
            result = att.Value;
            return true;
        

        var nodes = _root.Elements(binder.Name);
        if (nodes.Count() > 1)
        
            result = nodes.Select(n => n.HasElements ? (object)new DynamicXml(n) : n.Value).ToList();
            return true;
        

        var node = _root.Element(binder.Name);
        if (node != null)
        
            result = node.HasElements || node.HasAttributes ? (object)new DynamicXml(node) : node.Value;
            return true;
        

        return true;
    

--编辑--

为了使它与 xml 命名空间一起工作,我添加了 RemoveNamespaces 方法。

public class DynamicXml : DynamicObject

    XElement _root;
    private DynamicXml(XElement root)
    
        _root = root;
    

    public static DynamicXml Parse(string xmlString)
    
        return new DynamicXml(RemoveNamespaces(XDocument.Parse(xmlString).Root));
    

    public static DynamicXml Load(string filename)
    
        return new DynamicXml(RemoveNamespaces(XDocument.Load(filename).Root));
    

    private static XElement RemoveNamespaces(XElement xElem)
    
        var attrs = xElem.Attributes()
                    .Where(a => !a.IsNamespaceDeclaration)
                    .Select(a => new XAttribute(a.Name.LocalName, a.Value))
                    .ToList();

        if (!xElem.HasElements)
        
            XElement xElement = new XElement(xElem.Name.LocalName, attrs);
            xElement.Value = xElem.Value;
            return xElement;
        

        var newXElem = new XElement(xElem.Name.LocalName, xElem.Elements().Select(e => RemoveNamespaces(e)));
        newXElem.Add(attrs);
        return newXElem;
    

    public override bool TryGetMember(GetMemberBinder binder, out object result)
    
        result = null;

        var att = _root.Attribute(binder.Name);
        if (att != null)
        
            result = att.Value;
            return true;
        

        var nodes = _root.Elements(binder.Name);
        if (nodes.Count() > 1)
        
            result = nodes.Select(n => n.HasElements ? (object)new DynamicXml(n) : n.Value).ToList();
            return true;
        

        var node = _root.Element(binder.Name);
        if (node != null)
        
            result = node.HasElements || node.HasAttributes ? (object)new DynamicXml(node) : node.Value;
            return true;
        

        return true;
    

【讨论】:

如果失败返回多个元素而只返回一个元素并且您在 foreach 中使用它,这将非常有用!我建议更改“result = new DynamicXml(node);”到“结果=新列表()新DynamicXml(节点);”这将返回一个包含单个元素的列表。 @L.B 元素和属性不一样。因此,为每个语法使用不同的语法更有意义。 @RezoMegrelidze 这是我的代码,我想要这样。随意根据您的需要编写自己的版本 令人印象深刻。这对于在响应中包含大量内容的 API 非常有用,而我只对 2 或 3 个字段(元素)感兴趣。唯一的事情是现在我会醒着想知道这个黑魔法到底是如何运作的。谢谢。 很好,但我发现了一个错误 - 对于具有属性的空元素,代码返回一个空字符串,而不是像这里 &lt;Student Name="Ivan" /&gt; 这样的元素。我将这个不断变化的 39 DynamicXml 类行修复为这个 result = node.HasElements || node.HasAttributes ? (object)new DynamicXml(node) : node.Value;

以上是关于使用动态将 XML 反序列化为对象的主要内容,如果未能解决你的问题,请参考以下文章

如何将 XML 反序列化为 C# 中的对象? [复制]

将 Xml 反序列化为对象时出错 - xmlns='' 不是预期的

将 Xml 反序列化为对象时出错 - xmlns='' 不是预期的

您可以将旧版本的 xml 反序列化为更新的结构吗

将 xml xmpp 消息反序列化为对象

将 ODATA xml 序列化/反序列化为 C# 对象