转: .Net 4.0 ExpandoObject 使用

原文链接:http://www.cnblogs.com/GaryFeng/archive/2010/07/03/1770517.html

本篇文章中就ExpandoObject的基本使用进行一些demo。我们几乎都知道dynamic特性是.net 4.0中一个主要的新特性,而ExpandoObject正是这样的一个动态的类型。ExpandoObject允许我们在实例化之后在运行时进行成员的增加、删除。下面我们来看下基本的使用:

Adding Members

1)实例化

如果需要延迟绑定的话,我们需要用dynamic来定义ExpandpObject的实例化变量,关于dynamic的使用,估计大家都很清楚了。

1.dynamic obj = new ExpandoObject();  

2)增加属性成员

 

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.    Console.WriteLine("Employee's name :{0} age:{1}", 09.        employee.FirstName + employee.LastName, 10.        employee.Age); 11.}

 

测试的结果:

转: .Net 4.0 ExpandoObject 使用

3)增加Method

在增加方法的时候,先把成员表现成delegate,先看一个没有参数的无返回值的方法:

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

 

测试结果:

转: .Net 4.0 ExpandoObject 使用

上面的例子中是段很简单的逻辑就是超过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.       }

我们看下运行的结果:

转: .Net 4.0 ExpandoObject 使用

 

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

 

我们可以看到测试结果:

转: .Net 4.0 ExpandoObject 使用

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

我们看看运行的结果:

转: .Net 4.0 ExpandoObject 使用

我们可以看到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());
}

我们看到输出结果:

转: .Net 4.0 ExpandoObject 使用

好像有点大功告成了,其实远没有这么简单。我们来考虑几个问题,首先如果出现重复的节点怎么办,比如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;
}

来看看测试的效果:

转: .Net 4.0 ExpandoObject 使用

 

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进行一些尝试。

 

转: .Net 4.0 ExpandoObject 使用 作者:Henllyee Cui
出处: http://henllyee.cnblogs.com/
本文版权归作者和博客园共有,欢迎转载,但未经作者同意必须保留此段声明。

转载于:https://www.cnblogs.com/GaryFeng/archive/2010/07/03/1770517.html

上一篇:确定C#中类的default()值


下一篇:LINQ之路21:LINQ to XML之生成X-DOM(Projecting)