设为首页收藏本站language 语言切换
查看: 1458|回复: 0
收起左侧

动态装载和使用类型

[复制链接]
发表于 2010-2-25 10:19:33 | 显示全部楼层 |阅读模式
<p ><ccid_nobr>Reflection提供诸如Microsoft Visual Basic.NET和JScript语言编译器使用的底层结构来实施隐性后绑定。绑定是定位与某一特定类型相对应的声明的过程。当这个过程发生在运行的时候,而不是编译的时候,它被称为后绑定。VisualBasic.NET使你可以在你的代码中使用隐性后绑定;VisualBasic.NET编译器调用helper 方法,使用Reflection获得对象类型。传递给helper 方法的参数 使适当的方法可以在运行时被调用。这些参数是调用方法(对象)的实例,被调用方法的名字(字符串),及传递给被调用方法的参数。(一个对象数组)。<br/><br/>在以下代码例子中, Visual Basic.NET编译器通过Reflection隐性地 来对一在编译时不知类型的对象调用方法。HelloWorld 类有一种 PrintHello 方法,可以打印出&quot;<b>Hello World</b>&quot; 及传递给<b>rintHello</b> 方法的一些文本。本例中<b>rintHello </b>方法 调用实际上是Type. InvokeMember ; Visual Basic 代码 允许<b>rintHello</b> 方法被调用,仿佛 对象的类型 (helloObj)在编译时就已经知道了(前期绑定),而不是在运行时(后绑定)。<br/><table width="580" border="1" cellspacing="0" cellpadding="0" bordercolorlight="black" bordercolordark="#ffffff"><tr><td bgcolor="e6e6e6" class="code">[Visual Basic]<br/>Imports System<br/>Module Hello<br/>Sub Main()<br/>' Set up variable.<br/>Dim helloObj As Object<br/>' Create the object.<br/>helloObj=new HelloWorld()<br/>' Invoke the print method as if it was early bound<br/>' even though it's really late bound.<br/>helloObj.PrintHello(&quot;Visual Basic Late Bound&quot;)<br/>End Sub<br/>End Module</td></tr></table><br/><b>自定义绑定</b><br/>Reflection除了可以隐性地被编译器用于后绑定,也可以在代码中显示使用,来完成后绑定。<br/><br/>common language runtime 支持多种编程语言,这些语言的绑定规则互不相同。在前绑定的情况下,代码生成器能完全控制绑定。然而,在使用Reflection的后绑定中,绑定必须由自定义绑定控制。Binder类提供成员选择与调用的自定义控制。 <br/><br/>使用自定义绑定, 您可以在运行时装载assembly,获得assembly中关于类型的信息,指明您索要的类型,并且调用方法,访问字段,或类型的属性。如果在编译时您不知道对象的类型,该技术就显得格外有用,比如,当对象类型依赖于用户输入时。以下例子中的代码显示了在HelloWorld.dll assembly 中,被动态使用Reflection调用的方法,第一个在VisualBasic.NET,第二个在C#中。<br/><table width="580" border="1" cellspacing="0" cellpadding="0" bordercolorlight="black" bordercolordark="#ffffff"><tr><td bgcolor="e6e6e6" class="code">[Visual Basic]<br/>' This class is deployed as an assembly consisting Hello World string.<br/>rivate m_helloWorld As String=&quot;HelloWorld&quot;<br/>' Default public constructor.<br/>ublic Sub New()<br/><br/>End Sub 'New<br/><br/>' Print &quot;Hello World&quot; plus thepassed text.<br/>ublic Sub PrintHello(txt As String)<br/>' Output to the Console.<br/>Console.WriteLine((m_helloWorld &amp; &quot;&quot; &amp; txt))<br/>End Sub<br/>End Class<br/><br/>Imports System<br/>Imports System.Reflection<br/>Module VisualBasicLateHello<br/>Sub Main()<br/>' Set up the variables.<br/>Dim assem as System.Reflection.Assembly<br/>Dim obj as Object<br/>Dim helloType as Type<br/>Dim printMethod as MethodInfo<br/>' Load the assembly to use.<br/>assem=System.Reflection.Assembly.Load(&quot;HelloWorld&quot;)<br/>' Get the type to use from the assembly.<br/>helloType=assem.GetType(&quot;HelloWorld&quot;)<br/>' Get the method to use from the type.<br/>printMethod=helloType.GetMethod(&quotrintHello&quot;)<br/>' Create an instance of the type.<br/>obj=Activator.CreateInstance(helloType)<br/>' Create an array to hold the arguments.<br/>Dim args(1) as Object<br/>' Set the arguments.<br/>args(0)=&quot;From Visual Basic Late Bound&quot;<br/>' Invoke the method.<br/>printMethod.Invoke(obj, args)<br/>End Sub<br/>End Module<br/></td></tr></table><br/>以下为C# 版:<br/><table width="580" border="1" cellspacing="0" cellpadding="0" bordercolorlight="black" bordercolordark="#ffffff"><tr><td bgcolor="e6e6e6" class="code">[C#]<br/>// This class is deployed as an assemblyconsisting of one DLL,<br/>// called HelloWorld.dll.<br/>using System;<br/>public class HelloWorld {<br/>// Constant Hello World string.<br/>private const String m_helloWorld=&quot;HelloWorld&quot;;<br/>// Default public constructor.<br/>public HelloWorld() {<br/>}<br/>// Print &quot;Hello World&quot; plus thepassed text.<br/>public void PrintHello(String txt) {<br/>// Output to the Console.<br/>Console.WriteLine(m_helloWorld + &quot; &quot;+ txt);<br/>}<br/>}<br/><br/>// Illustrates reflection's late bindingfunctionality.<br/>// Calls the PrintHello method on a dynamicallyloaded<br/>// and created instance of the HelloWorldclass.<br/>using System;<br/>using System.Reflection;<br/>public class CSharpLateHello {<br/>public static void Main() {<br/>// Load the assembly to use.<br/>Assembly assem=Assembly.Load(&quot;HelloWorld&quot;);<br/>// Get the type to use from the assembly.<br/>Type helloType=assem.GetType(&quot;HelloWorld&quot;);<br/>// Get the method to call from the type.<br/>MethodInfo printMethod=helloType.GetMethod(&quotrintHello&quot;);<br/>// Create an instance of the HelloWorld class.<br/>Object obj=Activator.CreateInstance(helloType);<br/>// Create the args array.<br/>Object[] args=new Object[1];<br/>// Set the arguments.<br/>args[0]=&quot;From CSharp Late Bound&quot;;<br/>// Invoke the PrintHello method.<br/>printMethod.Invoke(obj, args);<br/>}<br/>}<br/></td></tr></table><br/><b>InvokeMember 与 CreateInstance</b><br/>可以使用Type.InvokeMember来调用某类型成员。各种类的CreateInstance 方法,例如System.Activator和System.Reflection.Assembly,是InvokeMember的专用形式,用于生成某类型新的实例。Binder类在这些方法中,被用于重载解析和参数转换。<br/><br/>以下例子中的代码显示了三种可能的参数转换及成员选择的组合。在Case1中, 不需要参数转换或成员选择。在Case 2中,只需要成员选择。在Case3中, 只需要参数转换。<br/><table width="580" border="1" cellspacing="0" cellpadding="0" bordercolorlight="black" bordercolordark="#ffffff"><tr><td bgcolor="e6e6e6" class="code">[C#]<br/>public class CustomBinderDriver<br/>{<br/>public static void Main (string[] arguments)<br/>{<br/>Type t=typeof (CustomBinderDriver);<br/>CustomBinder binder=new CustomBinder();<br/>BindingFlags flags=BindingFlags.InvokeMethod|BindingFlags.Instance|<br/>BindingFlags.Public|BindingFlags.Static;<br/><br/>//Case 1. Neither argument coercion nor memberselection is needed.<br/>args=new Object[] {};<br/>t.InvokeMember (&quotrintBob&quot;, flags,binder, null, args);<br/><br/>//Case 2. Only member selection is needed.<br/>args=new Object[] {42};<br/>t.InvokeMember (&quotrintValue&quot;, flags,binder, null, args);<br/><br/>//Case 3. Only argument coercion is needed.<br/>args=new Object[] {&quot;5.5&quot;};<br/>t.InvokeMember (&quotrintNumber&quot;,flags, binder, null, args);<br/>}<br/><br/>public static void PrintBob ()<br/>{<br/>Console.WriteLine (&quotrintBob&quot;);<br/>}<br/><br/>public static void PrintValue (long value)<br/>{<br/>Console.WriteLine (&quotrintValue ({0})&quot;,value);<br/>}<br/>public static void PrintValue (String value)<br/>{<br/>Console.WriteLine (&quotrintValue\&quot;{0}\&quot;)&quot;,value);<br/>}<br/><br/>public static void PrintNumber (double value)<br/>{<br/>Console.WriteLine (&quotrintNumber ({0})&quot;,value);<br/>}<br/>}<br/></td></tr></table><br/>当存在多于一个的同名成员时,就需要有重载解析。Binder.BindToMethod和Binder.BindToField 方法可以用来绑定到一个成员。Binder.BindToMethod也可以通过<b>get </b>和<b>set </b>属性访问器提供属性解析。<br/><br/><b>BindToMethod </b>返回可被调用的<b>MethodBase</b>. 如无可用的调用则返回<b>null</b>. 如果无法调用,BindToMethod 返回 MethodBase为 调用或 <b>null</b>。<b>MethodBase</b>返回值无需是match参数之一,尽管事实往往如此。<br/><br/>调用者 也许会想得到ByRef 参数的返回。所以,如果<b>BindTo</b>方法改动过参数数组,Binder 允许客户使参数数组映射回它原来的表格。为了实现这点,调用者必须确保参数顺序不变。当参数由名字传递,<b>Binder</b>重新整理参数组,以供调用者察看。<br/><br/>可用成员是指那些在类型或任何基本类型中定义的那些成员。如果指明BindingFlags.NonPublic,任何访问级别的成员都会在返回中。如果BindingFlags.NonPublic 没有被指明,binder必须执行访问规则。当指明<b>ublic或 NonPublic </b>绑定标志, 你必须也指明<b>Instance</b> 或<b>Static</b> 标志, 否则不会有成员返回。<br/><br/>如果只有一个成员与名字对应,就不需要回调,也就完成到这个方法的绑定。Case 1 中的代码例子表明了这一点:只有一个可用的<b>rintBob</b> 方法, 所以,不需要回调。<br/><br/>如在可用集中,有多于一个成员。所有这些方法被传递给<b>BindTo</b>方法, 再由它选择适当的方法,并且返回。在Case 2 中的代码例子中,有两种叫做<b>rintValue</b>的方法。合适的方法取决于对<b>BindToMethod</b>调用。<br/><br/><b>ChangeType </b>执行参数转换, 它把实际参数转变为选定方法的参数类型。即使类型已经完美匹配,ChangeType也会针对每个参数被调用。<br/><br/>在 Case 3 中的代码例子中, 值为&quot;5.5&quot;的<b>String</b>类型的一个实际参数以正式参数Double类型被传递给方法。要想调用成功,字符串值&quot;5.5&quot;必须被转变为一个<b>double</b>值。<b>ChangeType</b> 执行了这种转变。<br/><br/><b>ChangeType</b> 仅执行无损失转换, 如下表所示:<br/><table border="1" width="60%" class="tech"><tbody><tr><td width="50%"><b>Source Type</b></td><td width="50%"><b>Target Type</b></td></tr><tr><td>Any type</td><td>Its base type</td></tr><tr><td>Any type</td><td>Interface it implements</td></tr><tr><td>Char</td><td>UInt16, UInt32, Int32, UInt64, Int64, Single,Double</td></tr><tr><td>Byte</td><td>Char, UInt16, Int16, UInt32, Int32, UInt64,Int64, Single, Double</td></tr><tr><td>SByte</td><td>Int16, Int32, Int64, Single, Double</td></tr><tr><td>UInt16</td><td>UInt32, Int32, UInt64, Int64, Single, Double</td></tr><tr><td>Int16</td><td>Int32, Int64, Single, Double</td></tr><tr><td>UInt32</td><td>UInt64, Int64, Single, Double</td></tr><tr><td>Int32</td><td>Int64, Single, Double</td></tr><tr><td>UInt64</td><td>Single, Double</td></tr><tr><td>Int64</td><td>Single, Double</td></tr><tr><td>Single</td><td>Double</td></tr><tr><td>Non-reference type</td><td>Reference type</td></tr></tbody></table><br/>Type类有<b>Get</b>方法,可使用Binder类型的参数的来解析对某成员的引用。Type.GetConstructor,Type. GetMethod , 和 Type.GetProperty 通过提供某成员的签名信息来查找该成员。它们调用Binder.SelectMethod和Binder.SelectProperty 以选择适合方法的签名信息。<br/><br/>欢迎访问:微软开发者专栏(责任编辑 <ccid_nobr>尤北</ccid_nobr>)<p ></ccid_nobr><p align="center"></p></p>
您需要登录后才可以回帖 登录 | 论坛注册

本版积分规则

QQ|Archiver|手机版|小黑屋|sitemap|鸿鹄论坛 ( 京ICP备14027439号 )  

GMT+8, 2025-4-4 05:32 , Processed in 0.064924 second(s), 23 queries , Redis On.  

  Powered by Discuz!

  © 2001-2025 HH010.COM

快速回复 返回顶部 返回列表