详解C# 反射(Reflection)
C#反射(Reflection)
反射指程序可以访问、检测和修改它本身状态或行为的一种能力。
程序集包含模块,而模块包含类型,类型又包含成员。反射则提供了封装程序集、模块和类型的对象。
您可以使用反射动态地创建类型的实例,将类型绑定到现有对象,或从现有对象中获取类型。然后,可以调用类型的方法或访问其字段和属性。
优缺点
优点:
1、反射提高了程序的灵活性和扩展性。
2、降低耦合性,提高自适应能力。
3、它允许程序创建和控制任何类的对象,无需提前硬编码目标类。
缺点:
1、性能问题:使用反射基本上是一种解释操作,用于字段和方法接入时要远慢于直接代码。因此反射机制主要应用在对灵活性和拓展性要求很高的系统框架上,普通程序不建议使用。
2、使用反射会模糊程序内部逻辑;程序员希望在源代码中看到程序的逻辑,反射却绕过了源代码的技术,因而会带来维护的问题,反射代码比相应的直接代码更复杂。
反射(Reflection)的用途
反射(Reflection)有下列用途:
- 它允许在运行时查看特性(attribute)信息。
- 它允许审查集合中的各种类型,以及实例化这些类型。
- 它允许延迟绑定的方法和属性(property)。
- 它允许在运行时创建新类型,然后使用这些类型执行一些任务。
查看元数据
我们已经在上面的章节中提到过,使用反射(Reflection)可以查看特性(attribute)信息。
System.Reflection类的MemberInfo对象需要被初始化,用于发现与类相关的特性(attribute)。为了做到这点,您可以定义目标类的一个对象,如下:
System.Reflection.MemberInfoinfo=typeof(MyClass);
下面的程序演示了这点:
usingSystem; [AttributeUsage(AttributeTargets.All)] publicclassHelpAttribute:System.Attribute { publicreadonlystringUrl; publicstringTopic//Topic是一个命名(named)参数 { get { returntopic; } set { topic=value; } } publicHelpAttribute(stringurl)//url是一个定位(positional)参数 { this.Url=url; } privatestringtopic; } [HelpAttribute("InformationontheclassMyClass")] classMyClass { } namespaceAttributeAppl { classProgram { staticvoidMain(string[]args) { System.Reflection.MemberInfoinfo=typeof(MyClass); object[]attributes=info.GetCustomAttributes(true); for(inti=0;i当上面的代码被编译和执行时,它会显示附加到类MyClass上的自定义特性:
HelpAttribute
实例
在本实例中,我们将使用在上一章中创建的DeBugInfo特性,并使用反射(Reflection)来读取Rectangle类中的元数据。
usingSystem; usingSystem.Reflection; namespaceBugFixApplication { //一个自定义特性BugFix被赋给类及其成员 [AttributeUsage(AttributeTargets.Class| AttributeTargets.Constructor| AttributeTargets.Field| AttributeTargets.Method| AttributeTargets.Property, AllowMultiple=true)] publicclassDeBugInfo:System.Attribute { privateintbugNo; privatestringdeveloper; privatestringlastReview; publicstringmessage; publicDeBugInfo(intbg,stringdev,stringd) { this.bugNo=bg; this.developer=dev; this.lastReview=d; } publicintBugNo { get { returnbugNo; } } publicstringDeveloper { get { returndeveloper; } } publicstringLastReview { get { returnlastReview; } } publicstringMessage { get { returnmessage; } set { message=value; } } } [DeBugInfo(45,"ZaraAli","12/8/2012", Message="Returntypemismatch")] [DeBugInfo(49,"NuhaAli","10/10/2012", Message="Unusedvariable")] classRectangle { //成员变量 protecteddoublelength; protecteddoublewidth; publicRectangle(doublel,doublew) { length=l; width=w; } [DeBugInfo(55,"ZaraAli","19/10/2012", Message="Returntypemismatch")] publicdoubleGetArea() { returnlength*width; } [DeBugInfo(56,"ZaraAli","19/10/2012")] publicvoidDisplay() { Console.WriteLine("Length:{0}",length); Console.WriteLine("Width:{0}",width); Console.WriteLine("Area:{0}",GetArea()); } }//endclassRectangle classExecuteRectangle { staticvoidMain(string[]args) { Rectangler=newRectangle(4.5,7.5); r.Display(); Typetype=typeof(Rectangle); //遍历Rectangle类的特性 foreach(Objectattributesintype.GetCustomAttributes(false)) { DeBugInfodbi=(DeBugInfo)attributes; if(null!=dbi) { Console.WriteLine("Bugno:{0}",dbi.BugNo); Console.WriteLine("Developer:{0}",dbi.Developer); Console.WriteLine("LastReviewed:{0}", dbi.LastReview); Console.WriteLine("Remarks:{0}",dbi.Message); } } //遍历方法特性 foreach(MethodInfomintype.GetMethods()) { foreach(Attributeainm.GetCustomAttributes(true)) { DeBugInfodbi=(DeBugInfo)a; if(null!=dbi) { Console.WriteLine("Bugno:{0},forMethod:{1}", dbi.BugNo,m.Name); Console.WriteLine("Developer:{0}",dbi.Developer); Console.WriteLine("LastReviewed:{0}", dbi.LastReview); Console.WriteLine("Remarks:{0}",dbi.Message); } } } Console.ReadLine(); } } }当上面的代码被编译和执行时,它会产生下列结果:
Length:4.5
Width:7.5
Area:33.75
BugNo:49
Developer:NuhaAli
LastReviewed:10/10/2012
Remarks:Unusedvariable
BugNo:45
Developer:ZaraAli
LastReviewed:12/8/2012
Remarks:Returntypemismatch
BugNo:55,forMethod:GetArea
Developer:ZaraAli
LastReviewed:19/10/2012
Remarks:Returntypemismatch
BugNo:56,forMethod:Display
Developer:ZaraAli
LastReviewed:19/10/2012
Remarks:以上就是详解C#反射(Reflection)的详细内容,更多关于C#反射(Reflection)的资料请关注毛票票其它相关文章!