一、反射的定义:

审查元数据并收集关於它的类型信息的能力,元数据(编辑后的基本数据单元)就是一大堆表,编译器会创建一个类定义表,一个字段定义表,一个方法定义表等, System.Reflection 命名空间包含的几个类,允许你反射(解析)这些元数据的代码。

二、反射的作用:

动态的创建类型的实例,将类型邦定到现有对象,或从现有对象中获取类型。应用程序需要在运行时从某个特定的程序集中载入一个特定的类型,以便实现某个任务时可以用到反射。反射主要应用於类库,这些类库需要知道一个类型的定义,以便提供更多的功能。

三、应用要点:

现实应用程序中很少使用到反射,使用反射动态邦定需要牺牲性能,有些元数据信息是不能通过反射获取的,某些反射类型是专门为那些CLR开发编辑器开发使用的,所以你要意思到不是所有反射类型都是可以使用的。

四、取得Assembly的方法

Assembly.Load Assembly.LoadFile Assembly.LoadFrom Type对象的Assembly方法

五、反射的成员:

(1)MemberInfo-成员;  (2)ConstructorInfo-结构;  (3)FieldInfo-字段;  (4)MethodInfo-方法;  (5)PropertyInfo-属性;  (6)EventInfo-事件。

六、根据反射取得对象的Member信息

private void WriteReflectionInfo()
{
Type testType = typeof(Test);
Assembly assembly = testType.Assembly;
Response.Write("Assembly:" + assembly.FullName + "<br/>");
Type[] typeList = assembly.GetTypes(); // 获取类型
// 针对每个类型获取详细信息
foreach (Type type in typeList)
{
Response.Write("------------------------" + type.Namespace + type.Name + "------------------------<br/>");
// 获得类型的结构信息
ConstructorInfo[] constructs = type.GetConstructors(); // 获得类型的字段信息
FieldInfo[] fields = type.GetFields();
Response.Write("<b>类的公共字段信息如下:</b>" + "<br/>");
int a1 = ;
foreach (FieldInfo field in fields)
{
Response.Write((a1++).ToString() + ". " + field.Name + "<br/>");
} // 获得方法信息
MethodInfo[] methods = type.GetMethods(); Response.Write("<b>类的公共方法如下:</b>" + "<br/>");
int a2 = ;
foreach (MethodInfo method in methods)
{
ParameterInfo[] parameters = method.GetParameters();
ParameterInfo reparam = method.ReturnParameter;
Response.Write((a2++).ToString() + ". " + reparam.ParameterType.Name + " " + method.Name + "(");
int index = ;
foreach (ParameterInfo para in parameters)
{
if (index++ < parameters.Length - )
Response.Write(para.ParameterType.Name + " " + para.Name + ",");
else
Response.Write(para.ParameterType.Name + " " + para.Name);
}
Response.Write(")<br/>");
} // 获得属性的信息
PropertyInfo[] propertys = type.GetProperties(); Response.Write("<b>类的公共属性如下:</b>" + "<br/>");
int a3 = ;
foreach (PropertyInfo pro in propertys)
{
Response.Write((a3++).ToString() + ". " + pro.PropertyType.Name + " " + pro.Name + "{");
if (pro.CanRead) Response.Write("get;");
if (pro.CanWrite) Response.Write("set;");
Response.Write("}<br/>");
}
// 获得事件信息
EventInfo[] events = type.GetEvents(); Response.Write("<b>类的成员如下:</b>" + "<br/>");
// 获得成员
int a4 = ;
foreach (MemberInfo mi in type.GetMembers())
{
Response.Write((a4++).ToString() + ". " + mi.MemberType.ToString() + " : " + mi.Name + "<br/>");
}
}

七、动态创建对象

Assembly对象的 CreateInstance方法
Activator. CreateInstance方法
Type对象的 InvokeMember方法
// 使用Assembly的CreateInstance方法来取得对象的实例
private void Assembly_CreateInstance()
{
string assemblyName = "SqlModel";
string className = assemblyName + ".Member";
// 创建无参数实例
IDAL.IMember member = (IDAL.IMember)Assembly.Load(assemblyName).CreateInstance(className);
Response.Write("创建无参数实例:" + member.ID + "<br/>");
// 创建有参数实例
Object[] parameters = new Object[];
parameters[] = ;
IDAL.IMember member1 = (IDAL.IMember)Assembly.Load(assemblyName).CreateInstance(className, false, BindingFlags.Default, null, parameters, null, null);
Response.Write("创建有参数实例:" + member1.ID + "<br/>");
} // 使用Activator的CreateInstance方法来取得对象的实例
private void Activator_CreateInstance()
{
string assemblyName = "SqlModel";
string className = assemblyName + ".Member";
// 创建无参数实例
System.Runtime.Remoting.ObjectHandle obj = Activator.CreateInstance(assemblyName, className);
IDAL.IMember member = (IDAL.IMember)obj.Unwrap();
Response.Write("创建无参数实例:" + member.ID + "<br/>");
// 创建有参数实例
Object[] parameters = new Object[];
parameters[] = ;
System.Runtime.Remoting.ObjectHandle obj1 = Activator.CreateInstance(assemblyName, className, false, BindingFlags.CreateInstance, null, parameters, null, null, null);
IDAL.IMember member1 = (IDAL.IMember)obj1.Unwrap();
Response.Write("创建有参数实例:" + member1.ID + "<br/>");
} // 使用Type的InvokeMember方法来取得对象的实例
private void Type_InvokeMember()
{
string assemblyName = "SqlModel";
string className = assemblyName + ".Member";
Assembly assem = Assembly.Load(assemblyName);
Type type = assem.GetType(className); // 註意这里如果使用Type.GetType来取得Type的话,那麼assemblyName指定的类一定要是强命名的
// 创建无参数实例
IDAL.IMember member = (IDAL.IMember)type.InvokeMember(className, BindingFlags.CreateInstance, null, null, null);
Response.Write("创建无参数实例:" + member.ID + "<br/>");
// 创建有参数实例
Object[] parameters = new Object[];
parameters[] = ;
IDAL.IMember member1 = (IDAL.IMember)type.InvokeMember(className, BindingFlags.CreateInstance, null, null, parameters);
Response.Write("创建有参数实例:" + member1.ID + "<br/>");
}

八、动态调用对象方法

Type对象的 InvokeMember方法
MethodInfo对象的Invoke方法
// Type对象的 InvokeMember方法来动态调用方法
private void InvokeMember()
{
string assemblyName = "SqlModel";
string className = assemblyName + ".Member";
string methodName = String.Empty;
string result = String.Empty;
Assembly assem = Assembly.Load(assemblyName);
Object obj = assem.CreateInstance(className);
Type type = assem.GetType(className); // 註意这里如果使用Type.GetType来取得Type的话,那麼assemblyName指定的类一定要是强命名的
// 动态调用无参数的方法
methodName = "GetName";
result = (string)type.InvokeMember(methodName, BindingFlags.InvokeMethod, null, obj, null);
Response.Write(methodName + "方法的返回值:" + result + "<br/>");
// 动态调用有参数的方法
methodName = "Update";
Object[] methodParams = new Object[];
methodParams[] = DateTime.Now;
result = (string)type.InvokeMember(methodName, BindingFlags.InvokeMethod, null, obj, methodParams);
Response.Write(methodName + "方法的返回值:" + result + "<br/>");
// 动态调用参数构架函数的带有参数的方法
Object[] parameters = new Object[];
parameters[] = ;
obj = assem.CreateInstance(className,false,BindingFlags.CreateInstance, null, parameters, null, null);
result = (string)type.InvokeMember(methodName, BindingFlags.InvokeMethod, null, obj, methodParams);
Response.Write(methodName + "方法的返回值:" + result + "<br/>");
} // MethodInfo对象的Invoke方法来动态调用方法 private void MethodInfo_Invoke()
{
string assemblyName = "SqlModel";
string className = assemblyName + ".Member";
string methodName = String.Empty;
string result = String.Empty; Assembly assem = Assembly.Load(assemblyName);
Object obj = assem.CreateInstance(className);
Type type = assem.GetType(className); // 註意这里如果使用Type.GetType来取得Type的话,那麼assemblyName指定的类一定要是强命名的
// 动态调用无参数的方法
methodName = "GetName";
MethodInfo methodInfo = type.GetMethod(methodName);
result = (string)methodInfo.Invoke(obj, null);
Response.Write(methodName + "方法的返回值:" + result + "<br/>");
// 动态调用有参数的方法
methodName = "Update";
Object[] methodParams = new Object[];
methodParams[] = DateTime.Now;
MethodInfo method = type.GetMethod(methodName);
result = (string)method.Invoke(obj, methodParams);
Response.Write(methodName + "方法的返回值:" + result + "<br/>");
} --------------------------------------------------------------------------------
以上所使用的SqlModel.Member為:
新建一个SqlModel类库,在其下建立一个Member的类 namespace SqlModel
{
public class Member : IDAL.IMember
{
private int _id = ;
public int ID
{
get { return _id; }
set { _id = value; }
}
private string _name = "limin";
public string Name
{
get { return _name; }
set { _name = value; }
} public Member() { }
public Member(int id)
{
_id = id;
} private void Init()
{ } public string GetName()
{
return _name;
}
public string Update (DateTime cdate)
{
return "{" + String.Format("ID:{0},Name:{1},CreateDate:{2}",_id,_name,cdate) + "}";
}
}
}

C# 之 反射的更多相关文章

  1. 隐私泄露杀手锏 —— Flash 权限反射

    [简版:http://weibo.com/p/1001603881940380956046] 前言 一直以为该风险早已被重视,但最近无意中发现,仍有不少网站存在该缺陷,其中不乏一些常用的邮箱.社交网站 ...

  2. Java学习之反射机制及应用场景

    前言: 最近公司正在进行业务组件化进程,其中的路由实现用到了Java的反射机制,既然用到了就想着好好学习总结一下,其实无论是之前的EventBus 2.x版本还是Retrofit.早期的View注解框 ...

  3. 关于 CSS 反射倒影的研究思考

    原文地址:https://css-tricks.com/state-css-reflections 译者:nzbin 友情提示:由于演示 demo 的兼容性,推荐火狐浏览.该文章篇幅较长,内容庞杂,有 ...

  4. 编写高质量代码:改善Java程序的151个建议(第7章:泛型和反射___建议106~109)

    建议106:动态代理可以使代理模式更加灵活 Java的反射框架提供了动态代理(Dynamic Proxy)机制,允许在运行期对目标类生成代理,避免重复开发.我们知道一个静态代理是通过主题角色(Prox ...

  5. 运用Mono.Cecil 反射读取.NET程序集元数据

    CLR自带的反射机智和API可以很轻松的读取.NET程序集信息,但是不能对程序集进行修改.CLR提供的是只读的API,但是开源项目Mono.Cecil不仅仅可以读取.NET程序集的元数据,还可以进行修 ...

  6. .NET面试题系列[6] - 反射

    反射 - 定义,实例与优化 在面试中,通常会考察反射的定义(操作元数据),可以用反射做什么(获得程序集及其各个部件),反射有什么使用场景(ORM,序列化,反序列化,值类型比较等).如果答得好,还可能会 ...

  7. .NET基础拾遗(4)委托、事件、反射与特性

    Index : (1)类型语法.内存管理和垃圾回收基础 (2)面向对象的实现和异常的处理基础 (3)字符串.集合与流 (4)委托.事件.反射与特性 (5)多线程开发基础 (6)ADO.NET与数据库开 ...

  8. C++的性能C#的产能?! - .Net Native 系列五:.Net Native与反射

    此系列系小九的学堂原创翻译,翻译自微软官方开发向导,一共分为六个主题.本文是第五个主题:.Net Native与反射. 向导文链接:<C++的性能C#的产能?! - .Net Native 系列 ...

  9. [源码]Literacy 快速反射读写对象属性,字段

    Literacy 说明 Literacy使用IL指令生成方法委托,性能方面,在调用次数达到一定量的时候比反射高很多 当然,用IL指令生成一个方法也是有时间消耗的,所以在只使用一次或少数几次的情况,不但 ...

  10. SI与EMI(一) - 反射是怎样影响EMI

    Mark为期两天的EMC培训中大概分成四个时间差不多的部分,简单来说分别是SI.PI.回流.屏蔽.而在信号完整性的书籍中,也会把信号完整性分为:1.信号自身传输的问题(反射,损耗):2.信号与信号之间 ...

随机推荐

  1. 安装php5.5

    安装php5.5 ./configure --prefix=/usr/local/php5.5.14/ --with-apxs2=/usr/local/apache2.2.27/bin/apxs -- ...

  2. iOS获取文件和文件夹大小

    //通常用于删除缓存的时,计算缓存大小 //单个文件的大小 - (long long) fileSizeAtPath:(NSString*) filePath{ NSFileManager* mana ...

  3. python 自定义函数

    200 ? "200px" : this.width)!important;} --> 介绍 在Python中,定义一个函数要使用def语句,依次写出函数名.括号.括号中的参 ...

  4. C:常量、变量 、 表达式 、运算符、 枚举

    常量 变量 表达式 运算符 枚举 1.布尔类型只有真和假 2运算符 >,<,<=,>=,==,!=.判断两个数是否相等要使用双等号‘==’.逻辑运算符的表达式结果非真即假,&a ...

  5. C#之 HashSet(临时笔记,未参考资料,请慎重)

    HashSet是一个集合,类似于DataSet,但是其主要用途是用来存放同一种类型的元素(string.row.table等),如果添加的元素跟定义时初始的类型不一致,就会直接编译失败. 例如: Ha ...

  6. linux下的块设备驱动(一)

    块设备的驱动比字符设备的难,这是因为块设备的驱动和内核的联系进一步增大,但是同时块设备的访问的几个基本结构和字符还是有相似之处的. 有一句话必须记住:对于存储设备(硬盘~~带有机械的操作)而言,调整读 ...

  7. Java学习笔记之深入理解引用

    引言:Java中数据传递的方式,除了基本数据类型是按照值传递,其它类型全部是按照引用传递,这和C++有很大区别,但是很多网上文章都解释的不清楚,甚至是错误的,在查阅资料之后,下面整理出一个比较容易理解 ...

  8. 拖动窗体FormBorderStyle属性为None的窗体移动

    winform窗体的样式很单一,不够漂亮,往往我们需要对窗体进行重写,但是我们又要保留在重写前窗体本身带的功能,例如拖动窗体的头进行移动之类的. 一下方式可以实现该方法: [DllImport(&qu ...

  9. 创建C#串口通信程序详解

    在.NET平台下创建C#串口通信程序,.NET 2.0提供了串口通信的功能,其命名空间是System.IO.Ports.这个新的框架不但可以访问计算机上的串口,还可以和串口设备进行通信.我们将使用标准 ...

  10. UVA1099----Sharing Chocolate----在集合上的DP

    题目地址:http://uva.onlinejudge.org/index.php?option=com_onlinejudge&Itemid=8&page=show_problem& ...