C# 之 反射
一、反射的定义:
审查元数据并收集关於它的类型信息的能力,元数据(编辑后的基本数据单元)就是一大堆表,编译器会创建一个类定义表,一个字段定义表,一个方法定义表等, 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# 之 反射的更多相关文章
- 隐私泄露杀手锏 —— Flash 权限反射
[简版:http://weibo.com/p/1001603881940380956046] 前言 一直以为该风险早已被重视,但最近无意中发现,仍有不少网站存在该缺陷,其中不乏一些常用的邮箱.社交网站 ...
- Java学习之反射机制及应用场景
前言: 最近公司正在进行业务组件化进程,其中的路由实现用到了Java的反射机制,既然用到了就想着好好学习总结一下,其实无论是之前的EventBus 2.x版本还是Retrofit.早期的View注解框 ...
- 关于 CSS 反射倒影的研究思考
原文地址:https://css-tricks.com/state-css-reflections 译者:nzbin 友情提示:由于演示 demo 的兼容性,推荐火狐浏览.该文章篇幅较长,内容庞杂,有 ...
- 编写高质量代码:改善Java程序的151个建议(第7章:泛型和反射___建议106~109)
建议106:动态代理可以使代理模式更加灵活 Java的反射框架提供了动态代理(Dynamic Proxy)机制,允许在运行期对目标类生成代理,避免重复开发.我们知道一个静态代理是通过主题角色(Prox ...
- 运用Mono.Cecil 反射读取.NET程序集元数据
CLR自带的反射机智和API可以很轻松的读取.NET程序集信息,但是不能对程序集进行修改.CLR提供的是只读的API,但是开源项目Mono.Cecil不仅仅可以读取.NET程序集的元数据,还可以进行修 ...
- .NET面试题系列[6] - 反射
反射 - 定义,实例与优化 在面试中,通常会考察反射的定义(操作元数据),可以用反射做什么(获得程序集及其各个部件),反射有什么使用场景(ORM,序列化,反序列化,值类型比较等).如果答得好,还可能会 ...
- .NET基础拾遗(4)委托、事件、反射与特性
Index : (1)类型语法.内存管理和垃圾回收基础 (2)面向对象的实现和异常的处理基础 (3)字符串.集合与流 (4)委托.事件.反射与特性 (5)多线程开发基础 (6)ADO.NET与数据库开 ...
- C++的性能C#的产能?! - .Net Native 系列五:.Net Native与反射
此系列系小九的学堂原创翻译,翻译自微软官方开发向导,一共分为六个主题.本文是第五个主题:.Net Native与反射. 向导文链接:<C++的性能C#的产能?! - .Net Native 系列 ...
- [源码]Literacy 快速反射读写对象属性,字段
Literacy 说明 Literacy使用IL指令生成方法委托,性能方面,在调用次数达到一定量的时候比反射高很多 当然,用IL指令生成一个方法也是有时间消耗的,所以在只使用一次或少数几次的情况,不但 ...
- SI与EMI(一) - 反射是怎样影响EMI
Mark为期两天的EMC培训中大概分成四个时间差不多的部分,简单来说分别是SI.PI.回流.屏蔽.而在信号完整性的书籍中,也会把信号完整性分为:1.信号自身传输的问题(反射,损耗):2.信号与信号之间 ...
随机推荐
- javascript里面dom操作和兼容问题汇总
DOM 增,删,改,查 oUl.children 获取UL的子节点 有length的属性 oUl.children[0] 获取UL第1个子节点 使 ...
- hibernate里createSQLQuery的addEntity()和setResultTransformer()方法
http://langgufu.iteye.com/blog/1565397 ————————————————————————————————————————————————————————————— ...
- Android问题-XE5提示"[DCC Fatal Error] Project1.dpr(1): F1027 Unit not found: 'System.pas' or binary equivalents (.dcu/.o)"
问题现象:Checking project dependencies...Compiling Project1.dproj (Debug, Android)dcc command line for & ...
- UVALive 3953 Strange Billboard (状态压缩+枚举)
Strange Billboard 题目链接: http://acm.hust.edu.cn/vjudge/contest/129733#problem/A Description The marke ...
- SpriteParticle II
[SpriteParticle II] 1.Randomizing the Starting Position 2.Setting the Initial Angle 3.Setting a Part ...
- 【Android框架进阶〖0〗】ThinkAndroid注解机制
由于项目需要,开始研究ThinkAndroid. 个人认为该框架的注解机制十分新颖,所以先研究这个,顺便学习下 Java 的annotation. 粗略的看了看,该机制在BaseActivity中初始 ...
- Heritrix 3.1.0 源码解析(三十七)
今天有兴趣重新看了一下heritrix3.1.0系统里面的线程池源码,heritrix系统没有采用java的cocurrency包里面的并发框架,而是采用了线程组ThreadGroup类来实现线程池的 ...
- 26.怎样在Swift中定义宏?
Swift 中没有宏定义,苹果建议使用let 或者 get 属性来替代宏定义值.虽然没有#define,但我们仍然可以使用 #if 并配合编译的配置来完成条件编译.下面会列出Swift项目开发中的一些 ...
- StringUtils 字符串工具类
package com.thinkgem.jeesite.common.utils; import java.io.File; import java.io.IOException; import j ...
- JavaScript谁动了你的代码
到目前为止,同学你知道了JavaScript的历史,也了解其"你想是啥就是啥"的变量系统.相信凭借你深厚的Java或者C++功底,再加上程序员特有的自傲气质,你肯定会信心满满:自信 ...