【IT168技术】最近的一个项目用到很多不同结构的XML文件. 于是就在网上搜索了一些文章, 结合实际遇到的问题写成自己要的代码.
目标
基于已经有的XML文件,例如:
<root>
<books>
<book>
<author>John Savacki</author>
<title>E.G.Title</title>
<price>20.50</price>
</book>
<book>
<author>Tom Curly</author>
<title>E.G.Title 2</title>
<price>26.50</price>
</book>
</books>
</root>
<books>
<book>
<author>John Savacki</author>
<title>E.G.Title</title>
<price>20.50</price>
</book>
<book>
<author>Tom Curly</author>
<title>E.G.Title 2</title>
<price>26.50</price>
</book>
</books>
</root>
或者是类似任意结构的XML文件, 可以实现初始化
dynamic dx = new DynamicXml(xml);
并且读取属性值:
dx.books.book[0].author, dx.books.book[2].price
解决方案
自然是选择C# 4.0 中引入的 DynamicObject, 关键是已有的.net框架中不能提供这样的功能.那么就从这个类中继承生成子类, 同时也需要实现 IEnumerable.
稍微介绍一下:DynamicObject 类使您能够定义可以动态对象上执行哪些操作以及如何执行这些操作。 如果只需要设置和获取属性的操作,您可以只覆盖 TrySetMember 和 TryGetMember 方法。IEnumerable 公开枚举器,该枚举器支持在非泛型集合上进行简单迭代。
应该大家都知道要实现自定义集合的IEnumerable 就要完成
public IEnumerator GetEnumerator()
也就是说, 为了实现需求, 至少我们需要完成三个函数:
TrySetMember, TryGetMember, GetEnumerator
实现
通过上面的分析, 我们知道这个类看起来应该是这样:
public class DynamicXml : DynamicObject, IEnumerable
{
public DynamicXml(string text)
{
}
public override bool TryGetMember(GetMemberBinder binder, out object result)
{
}
public override bool TryGetIndex(GetIndexBinder binder, object[] indexes, out object result)
{
}
public IEnumerator GetEnumerator()
{
}
}
{
public DynamicXml(string text)
{
}
public override bool TryGetMember(GetMemberBinder binder, out object result)
{
}
public override bool TryGetIndex(GetIndexBinder binder, object[] indexes, out object result)
{
}
public IEnumerator GetEnumerator()
{
}
}
在构造函数里我们希望可以读取XML的内容, 加载所有元素的值.
在TrySetMember中, 我们为设置成员值的操作提供实现, 以便为设置属性值指定动态行为。在本文中, 我们只考虑读取XML内容, 所以不需要重写这个函数.
在TryGetMember中, 我们要为获取成员值的操作提供实现。
在 TryGetIndex中, 我们要为按索引获取值的操作提供实现.
实现的代码如下:
public class DynamicXml : DynamicObject, IEnumerable{
private readonly List<XElement> _elements;
public DynamicXml(string text)
{
var doc = XDocument.Parse(text);
_elements = new List<XElement> { doc.Root };
}
protected DynamicXml(XElement element)
{
_elements = new List<XElement> { element };
}
protected DynamicXml(IEnumerable<XElement> elements)
{
_elements = new List<XElement>(elements);
}
private readonly List<XElement> _elements;
public DynamicXml(string text)
{
var doc = XDocument.Parse(text);
_elements = new List<XElement> { doc.Root };
}
protected DynamicXml(XElement element)
{
_elements = new List<XElement> { element };
}
protected DynamicXml(IEnumerable<XElement> elements)
{
_elements = new List<XElement>(elements);
}