转: .Net 4.0 ExpandoObject 使用
本篇文章中就ExpandoObject的基本使用进行一些demo。我们几乎都知道dynamic特性是.net 4.0中一个主要的新特性,而ExpandoObject正是这样的一个动态的类型。ExpandoObject允许我们在实例化之后在运行时进行成员的增加、删除。下面我们来看下基本的使用:
Adding Members
1)实例化
如果需要延迟绑定的话,我们需要用dynamic来定义ExpandpObject的实例化变量,关于dynamic的使用,估计大家都很清楚了。
2)增加属性成员
02.public void ExpandoObjectTest() 03.{ 04. dynamic employee = new ExpandoObject(); 05. employee.FirstName = "Henry"; 06. employee.LastName = "Cui"; 07. employee.Age = 23; 08. Console.WriteLine("Employee's name :{0} age:{1}", 09. employee.FirstName + employee.LastName, 10. employee.Age); 11.}
测试的结果:
3)增加Method
在增加方法的时候,先把成员表现成delegate,先看一个没有参数的无返回值的方法:
02.public void ExpandoObjectTest() 03.{ 04. dynamic employee = new ExpandoObject(); 05. employee.FirstName = "Henry"; 06. employee.LastName = "Cui"; 07. employee.Age = 23; 08. employee.SayHello = (Action)(() => 09. { 10. Console.WriteLine("{0} say \"Hello\" at {1}", 11. employee.FirstName+" "+employee.LastName, 12. DateTime.UtcNow.ToString()); 13. }); 14. employee.SayHello(); 15.}在上面的示例中我们将SayHello定义成委托Action类型,并给了默认方法。
下面来看一个有参数、有返回值的:
01.[TestMethod] 02. public void ExpandoObjectTest() 03. { 04. dynamic employee = new ExpandoObject(); 05. employee.FirstName = "Henry"; 06. employee.LastName = "Cui"; 07. employee.Age = 23; 08. employee.SayHello = (Action)(() => 09. { 10. Console.WriteLine("{0} say \"Hello\" at {1}", 11. employee.FirstName+" "+employee.LastName, 12. DateTime.UtcNow.ToString()); 13. }); 14. 15. employee.GetSalary = (Func<int, decimal>)((month) => 16. { 17. if (month > 8) 18. return 5000; 19. return 4000; 20. }); 21. Console.WriteLine("The employee's october salary is :${0}", 22. employee.GetSalary(10).ToString()); 23. 24. }
测试结果:
上面的例子中是段很简单的逻辑就是超过8月份的时候就返回$5000,呵呵。
4)增加Event
在实例中我们定义一个请假事件,员工请假就会上报给经理:
01.[TestClass] 02. public class DynamicTest 03. { 04. [TestMethod] 05. public void ExpandoObjectTest() 06. { 07. dynamic employee = new ExpandoObject(); 08. employee.FirstName = "Henry"; 09. employee.LastName = "Cui"; 10. employee.Age = 23; 11. employee.SayHello = (Action)(() => 12. { 13. Console.WriteLine("{0} say \"Hello\" at {1}", 14. employee.FirstName+" "+employee.LastName, 15. DateTime.UtcNow.ToString()); 16. }); 17. employee.GetSalary = (Func<int, decimal>)((month) => 18. { 19. if (month > 8) 20. return 5000; 21. return 4000; 22. }); 23. employee.AskForLeaveEvent = null; 24. employee.AskForLeaveEvent += new EventHandler(OnEmployeeLeave); 25. employee.AskForLeaveEvent(employee,new EventArgs()); 26. } 27. public void OnEmployeeLeave(object sender, EventArgs e) 28. { 29. dynamic em = (dynamic)sender; 30. Console.WriteLine("Report Manager:{0} is asking for leave", em.FirstName + " " + em.LastName); 31. }我们看下运行的结果:
Remove Members
其实ExpandoObject继承了IDictionary<String, Object>的接口,所以我们枚举出在运行时增加的那些成员.
1)枚举出已经存在的成员
我们就来枚举出刚才在employee中增加的成员们:
1.foreach (var pro in (IDictionary<string, Object>)employee) 2.{ 3. Console.WriteLine(pro.Key+" "+pro.Value); 4.}
我们可以看到测试结果:
2)移除成员
其实我们还是利用了ExpandoObject实现了IDictionary接口去实现的,我们移除掉AskForLeaveEvent事件:
1.((IDictionary<string, object>)employee).Remove("AskForLeaveEvent"); 2.foreach (var pro in (IDictionary<string, Object>)employee) 3.{ 4. Console.WriteLine(pro.Key+" "+pro.Value); 5.}我们看看运行的结果:
我们可以看到AskForLeaveEvent被移除了。
总结
本文中主要介绍了ExpandoObject的基本使用,我们发现真的有点动态语言的风味,写过javascript的人感觉会太别爽,呵呵。下文中会就ExpandoObject的原理以及一些扩展就行一些说明。
本篇文章就ExpandoObject的一些高级的使用进行一些示例。
例子
首先要说的一点,为什么我们在定义动态类型的ExpandoObject时,必须要使用dynamic关键字呢,因为如果我们使用ExpandoObject 进行定义时,那么我们定义的变量就是一个静态类型ExpandoObject的实例化。
下面我们来做个例子就是如何将xml的表示成面向对象的形式。其实在c#3.0中已经提供了Linq To Xml的方式让我们来操作xml,确实比以前的dom方式方便了很多,但是觉得还是看着不太优雅。我们先来看一个Linq To Xml的示例:
private XElement CreateByXelement()
{
var xelement = new XElement(
"Employee",
new XElement("FirstName","Henry"),
new XElement("LastName","Cui"),
new XElement("Age",23),
new XElement(
"Company",
new XElement("Name","XXXX"),
new XElement("Address","Suzhou China")
)
);
return xelement;
}
这种方式我觉得是比以前的dom方式更为直观了,但是希望能够以更加优雅的方式来表示:
private dynamic CreateByExpandoObject()
{
dynamic employee = new ExpandoObject();
employee.FistName="Henry";
employee.LastName="Cui";
employee.Age=23;
employee.Company = new ExpandoObject();
employee.Company.Name="XXXX";
employee.Company.Address="Suzhou China";
return employee;
}
转换
也许现在最大的疑问就是想XElement一样提供了Save方法,这里我们来写些辅助的方法进行ExpandoObject到xml的转换吧:
private XElement ConvertExpandoObjectToXelement(string eleName, dynamic node)
{
var xNode = new XElement(eleName);
foreach (var pro in (IDictionary<string, object>)node)
{
if (pro.Value is ExpandoObject)
{
xNode.Add(ConvertExpandoObjectToXelement(pro.Key, pro.Value));
}
else
{
xNode.Add(new XElement(pro.Key, pro.Value));
}
}
return xNode;
}
来看看一个测试:
[TestMethod]
public void TestExpandoConvert()
{
var element = ConvertExpandoObjectToXelement("Employee",
CreateByExpandoObject());
Console.WriteLine(element.ToString());
}
我们看到输出结果:
好像有点大功告成了,其实远没有这么简单。我们来考虑几个问题,首先如果出现重复的节点怎么办,比如Employee受聘用。我们可以List结合来表示:
private dynamic CreateByExpandoObject()
{
dynamic employee = new ExpandoObject();
employee.FistName="Henry";
employee.LastName="Cui";
employee.Age=23;
employee.Company = new List<dynamic>();
employee.Company.Add(new ExpandoObject());
employee.Company[0].Name = "XXXX";
employee.Company[0].Address = "Suzhou China";
employee.Company.Add(new ExpandoObject());
employee.Company[1].Name = "YYYY";
employee.Company[1].Address = "Suzhou China";
return employee;
}
然后我们修改下转换增加对List<dynamic>类型的处理:
private XElement ConvertExpandoObjectToXelement(string eleName, dynamic node)
{
var xNode = new XElement(eleName);
foreach (var pro in (IDictionary<string, object>)node)
{
if (pro.Value is ExpandoObject)
{
xNode.Add(ConvertExpandoObjectToXelement(pro.Key, pro.Value));
}
else
{
if (pro.Value is List<dynamic>)
{
foreach (var child in (List<dynamic>)pro.Value)
{
xNode.Add(ConvertExpandoObjectToXelement(pro.Key, child));
}
}
else
{
xNode.Add(new XElement(pro.Key, pro.Value));
}
}
}
return xNode;
}
来看看测试的效果:
API的支持
1)查询
在Linq To Xml中提供了比如:Element、Elements、Descendant、Descendants方法来查询。
而对于我们使用ExpandoObject可以这样:对于单个的属性我们直接通过对象的属性去访问就可以得到了,对于List类型的我们可以使用Linq的语法:
var company = from c in (List<dynamic>)CreateByExpandoObject().Company
where c.Name == "XXXX"
select c;
Console.WriteLine(company.First().Name);
2)修改
对于简单属性的修改直接通过对象的属性就可以去修改了,而对于List<dyniamic>类型:
foreach (var child in (List<dynamic>)CreateByExpandoObject().Company)
{
if (child.Name == "XXXX")
{
child.Address = "Shanghai China";
}
}
总结
本文就如何使用ExpandoObject在xml领域中的使用进行了一些示例,只是一个初略的demo,不能说是解决方案。其实我们还可以通过dynamic库中的另外一个类型DynamicObject来实现,实现起来更为优雅,更为方便。在下文中会就DynamicObject操作xml进行一些尝试。
转自:http://henllyee.cnblogs.com/

浙公网安备 33010602011771号