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

47

是否可能像下面这样将未知的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)));

9
为什么不试试呢? - Roy Dictus
我尝试使用抛出异常的方式,但是同样的逻辑在Json中可以工作,为什么不支持XML呢? - user1875919
可能是将XML转换为动态C#对象的重复问题。 - Huseyin Yagli
1个回答

80

你可能想要尝试这个。

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

8
如果仅返回一个元素并且您在foreach中使用它,则此方法非常有效!我建议将"result = new DynamicXml(node);"更改为"result = new List<DynamicXml>() { new DynamicXml(node) };",这将返回一个只有一个元素的列表。 - Matthew M.
3
元素和属性并不相同,因此为每个元素和属性使用不同的语法更加合理。 - Rezo Megrelidze
13
这是我的代码,我希望它是这个样子的。根据您的需求随意编写您自己的版本。 - L.B
4
令人印象深刻。 这对于附加在API上的处理响应中有大量冗余信息,而我只对2或3个字段(元素)感兴趣非常有用。 唯一的问题是现在我会不安地想知道这种黑魔法是如何工作的。谢谢。 - AntDC
2
很好,但我发现一个错误 - 对于带有属性的空元素,代码返回一个空字符串而不是像这样的元素<Student Name="Ivan" />。我通过将39行的DynamicXml类更改为以下内容来修复此问题:result = node.HasElements || node.HasAttributes ? (object)new DynamicXml(node) : node.Value; - Vasyl Zvarydchuk
显示剩余6条评论

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接