C#反射的应用小结

1.何谓反射?

反射就是在运行的时候发现对象的相关信息.根据这些信息可以动态的执行对象的方法以及获取对象的属性所存储的值.使用.NET编写的代码时自动反射的,或者说是自我描述的.之所以可以反射,是通过编译后产生的元数据来做到的.因此,你可以在你的程序中使用反射来查找托管代码中的类型(包括类的名称,方法和参数)和与其相关的信息这其中包括执行被发现的代码.你也可以在程序运行的时候使用反射来创建,编译和运行代码.

2.反射的实例化

public static Assembly Load(string);

public static Assembly LoadFrom(string);

案例:

先定义一个类,方便以后使用:

using System;

using System.Collections.Generic;

using System.Linq;

using System.Reflection;

using System.Text;

using System.Threading.Tasks;

namespace reflecting

{

class Program

{

static void Main(string[] args)

{

Assembly ass = Assembly.Load("reflecting");

Type t = ass.GetType("reflecting.NewClassw");

object o = Activator.CreateInstance(t, "grayworm", "http://hi.baidu.com/grayworm");

MethodInfo mi = t.GetMethod("show");

mi.Invoke(o, null);

Console.ReadKey();

}

}

public class NewClassw

{

public NewClassw(string name, string sex)

{ }

public NewClassw()

{ }

public void show()

{

Console.WriteLine("hello,syx");

}

}

}

第一部分:找到特定的成员

列出程序集中类的名称:

Assembly myAssembly = Assembly.Load("Basics");
Type[] types = myAssembly.GetTypes();
foreach(Type type in types)
{
  if(type.IsClass)
     Console.WriteLine(type.Name);
}

直接访问某个类(假如访问的是SomeClass类中带一个参数的方法)

MethodInfo mi = typeof(SomeClass).GetMethod("SomeMethod");

过滤某些特定的成员

System.Type类也提供了一些方法,用于把包含在一个类里或者其它的类型里的特定的类型过滤到一个集合中。如GetConstructors方法,GetMethods方法,GetProperties方法和GetEvents方法均允许你以数组的方式返回所有给定的类型或者通过使用过滤条件只返回特定的类型集合。
以下是过滤出是公用和静态的方法:

MethodInfo[] mis = typeof(OtherClass).GetMethods(BindingFlags.Public | BindingFlags.Static);

注释:如果想得到私有成员,则用BindingFlags.NonPublic,但你需要有相应的权限.

搜索某些特定的成员

FieldInfo fi;

AnotherClass ac = new AnotherClass();

MemberInfo[] memInfo = ac.GetType().FindMembers(MemberTypes.Field, BindingFlags.NonPublic | BindingFlags.Instance, null, null);

foreach (MemberInfo m in memInfo)

{

fi = m as FieldInfo;

if (fi != null)

{

Console.WriteLine("{0} of value:{1}", fi.Name, fi.GetValue(ac));

}

}

自定义搜索

这个程序定义了一个委托(与委托MemberFilter具有相同签名的方法)MySearchDelegate,用于定制搜索条件。创建了一个类filterObject,其包含两个字段,辅助我们自定义搜索条件。程序中调用了FindMembers,指出我们需要所有的属性类型。当一个属性类型被发现时,程序将激发MySearchDelegate并且传给它一个filterCriteria实例对象,这个委托将判断成员的名称是否满足自定义的搜索条件来返回True还是False。

using System;

using System.Reflection;

namespace Basics2

{

/**/

/// <summary>

/// 被反射的类

/// </summary>

public class SomeClass

{

private int m_id;

public int ID

{

get { return this.m_id; }

set { this.m_id = value; }

}

private string m_name;

public string Name

{

get { return this.m_name; }

set { this.m_name = value; }

}

private int m_type;

public int Type

{

get { return this.m_type; }

set { this.m_type = value; }

}

}

/**/

/// <summary>

/// 自定义的过滤对象类

/// </summary>

public class filterObject

{

public string criterion1 = "Name";

public string criterion2 = "ID";

}

public class Basics

{

/**/

/// <summary>

/// 自定义的搜索条件,回调的方法

/// </summary>

/// <param name="memberInfo"></param>

/// <param name="filterCriteria"></param>

/// <returns></returns>

public static bool MySearchDelegate(MemberInfo memberInfo, object filterCriteria)

{

if (memberInfo.Name == ((filterObject)filterCriteria).criterion1 || memberInfo.Name == ((filterObject)filterCriteria).criterion2)

return true;

return false;

}

public static void Main()

{

PropertyInfo pi;

//绑定自定义的搜索条件

MemberFilter mf = new MemberFilter(MySearchDelegate);

SomeClass sc = new SomeClass();

//使用FindMembers返回指定的属性

MemberInfo[] memInfo = sc.GetType().FindMembers(MemberTypes.Property, BindingFlags.GetProperty | BindingFlags.Public | BindingFlags.Instance, mf, new filterObject());

foreach (MemberInfo m in memInfo)

{

pi = m as PropertyInfo;

Console.WriteLine(pi.Name);

}

Console.ReadLine();

}

}

}

第二部分  执行发现的代码

执行发现的代码的过程基本上要遵循以下几个基本的步骤:

1.加载程序集

2.找到你希望使用的类型或者类

3.创建该类型(或者类)的一个实例

4.找到你希望执行的该类型的某个方法

5.得到该方法的参数

6.调用该对象的方法并且传递给它恰当的参数.

创建发现的代码的实例

一旦找到你要找的类型,就可以使用System.Activator创建此类型的一个实例.你将会使用Avtivator类的方法CreateInstance众多版本中的一个.CreateInstance允许你指定你想要创建的对象,并且可选择的参数会应用到该对象的构造器上.

(1)在这里该对象的默认的构造器不用床底参数:

//SomeClass为一个类

object obj = System.Activator.CreateInstance(typeof(SomeClass));

(2)假设你想要创建一个特定对象的实例,其构造器是需要传递参数的.为此你需要把这些阐述的值作为一个数字的形式传递给CreateInstance.每一个参数的值需要对应该参数的类型,并且数组中参数的值需要与构造器的签名的顺序一致.

Type[] ts = {typeof(Int32)};
ConstructorInfo ci = typeof(SomeClass).GetConstructor(ts);
ParameterInfo[] pi = ci.GetParameters();
//创建一个数组,与返回的长度一样
object[] param = new object[pi.Length];
//给数组每个参数赋值
foreach(ParameterInfo p in pi)
{
  if(p.ParameterType == typeof(Int32))
    param[p.Position] = 100; //赋值
}
//最后可以创建实例化了
object o = System.Activator.CreateInstance(typeof(SomeClass),param);

到此,你已经得到了你的对象(SomeClass)的一个实例.接下来,让我们了解一下,如何调用该对象的方法.在之前,我们查询构造器的参数并把参数的值传给构造器,对于方法而言,这个处理过程是一样的.假设SomeClass类有一个SomeMethod方法,你想调用这个方法.为了保证粒子足够简单,建设方法SomeMethod没有任何参数(参数的处理过程同上.)为了能够用SomeMethod,你需要获取关于该方法的MethodInfo对象的一个引用.在这里你可以使用GetMethod或者GetMethods方法在你的类型上搜索.让我们使用GetMethod,并给其传递方法的名称.

MethodInfo mi=typeof(SomeClass).GetMethod(“SomeMethod”);

你不仅游泳了SomeClass的一个实例,而且也拥有了你希望调用该对象方法的引用mi,因此你可以使用MethodInfo.invoke调用你的目标方法了,你需要传递包含该方法的对象的会理和该方法需要的一组参数的值.如下:

mi.Invoke(o,null);

现在已经成功的创建了某个对象的一个实例,找到了该对象的某个方法,并且成功的调用了,而这些在涉及之初没有必要要知道该对象.

可以很容易的沿着这个例子向外延伸,创建一个类似于测试工具的实用工具.

C#高级编程----反射的小结的更多相关文章

  1. C#高级编程 反射 代码示例

    反射 反射(Reflection)是.NET中的重要机制,通过反射,可以在运行时获得.NET中每一个类型(包括类.结构.委托.接口和枚举等)的成员,包括方法.属性.事件,以及构造函数等. 还可以获得每 ...

  2. WebGL高级编程:开发Web3D图形 PDF(中文版带书签)

    WebGL高级编程:开发Web3D图形 目录 WebGL简介11.1 WebGL基础11.2 浏览器3D图形吸引人的原因21.3 设计一个图形API31.3.1 即时模式API31.3.2 保留模式A ...

  3. Atitit. 高级软件工程师and 普通的区别 高级编程的门槛总结

    Atitit.  高级软件工程师and 普通的区别 高级编程的门槛总结 1. 完备的知识体系 2 2. 编程理论/原理的掌握 2 1.1. 掌握常用的概念(ORM,IOC,AOP,event driv ...

  4. ASP.NET MVC5 高级编程 第3章 视图

    参考资料<ASP.NET MVC5 高级编程>第5版 第3章 视图 3.1 视图的作用 视图的职责是向用户提供界面. 不像基于文件的框架,ASP.NET Web Forms 和PHP ,视 ...

  5. ASP.NET MVC 4高级编程(第4版)

    <ASP.NET MVC 4高级编程(第4版)> 基本信息 作者: (美)Jon Galloway    Phil Haack    Brad Wilson    K. Scott All ...

  6. 《Hadoop》对于高级编程Hadoop实现构建企业级安全解决方案

    本章小结 ●    理解企业级应用的安全顾虑 ●    理解Hadoop尚未为企业级应用提供的安全机制 ●    考察用于构建企业级安全解决方式的方法 第10章讨论了Hadoop安全性以及Hadoop ...

  7. C# 6 与 .NET Core 1.0 高级编程 - 38 章 实体框架核心(上)

    译文,个人原创,转载请注明出处(C# 6 与 .NET Core 1.0 高级编程 - 38 章 实体框架核心(上)),不对的地方欢迎指出与交流. 章节出自<Professional C# 6 ...

  8. C# 6 与 .NET Core 1.0 高级编程 - 41 ASP.NET MVC(上)

    译文,个人原创,转载请注明出处(C# 6 与 .NET Core 1.0 高级编程 - 41 ASP.NET MVC(上)),不对的地方欢迎指出与交流. 章节出自<Professional C# ...

  9. C# 6 与 .NET Core 1.0 高级编程 - 41 ASP.NET MVC(中)

    译文,个人原创,转载请注明出处(C# 6 与 .NET Core 1.0 高级编程 - 41 ASP.NET MVC(中)),不对的地方欢迎指出与交流. 章节出自<Professional C# ...

随机推荐

  1. 关于报错stale element reference: element is not attached to the page document处理

    1.现象 在执行脚本时,有时候引用一些元素对象会抛出如下异常 org.openqa.selenium.StaleElementReferenceException: stale element ref ...

  2. Day6-------------ext4文件系统

    1.cp /etc/passwd /sdb6 把/etc/passwd的内容写入 sdb6 写入过程:日志------------>刷到硬盘 2.ext4已经有点过时 xfs:可存海量数据 bt ...

  3. bert 词典扩充方案

  4. IntelliJ IDEA使用教程

    注:本文来源:李学凯 的<IntelliJ IDEA使用教程 (总目录篇)> 一:(总目录篇)_1:硬件要求 IntelliJ IDEA 对硬件的要求看上去不是很高.可是实际在开发中其实并 ...

  5. Java 清理和垃圾回收

    java.lang.ref.cleaner包 finalize()//该方法已过时,有风险,慎用 1.对象不可能被垃圾回收 2.垃圾回收并不等于"析构" 只有当垃圾回收发生时fin ...

  6. spoj New Distinct Substrings

    vjudge原地爆炸... 题意:求一个字符串不同的子串的个数 策略:后缀数组 利用后缀数组的sa和height两个功能强大的数组,我们可以实现上述操作 首先有个很显然的结论:一个字符串的所有子串=它 ...

  7. 【图的遍历】广度优先遍历(DFS)、深度优先遍历(BFS)及其应用

    无向图满足约束条件的路径 •[目的]:掌握深度优先遍历算法在求解图路径搜索问题的应用 [内容]:编写一个程序,设计相关算法,从无向图G中找出满足如下条件的所有路径:  (1)给定起点u和终点v.  ( ...

  8. hdu 1240 3维迷宫 求起点到终点的步数 (BFS)

    题意,给出一个N,这是这个三空间的大小,然后给出所有面的状况O为空地,X为墙,再给出起始点的三维坐标和终点的坐标,输出到达的步数 比较坑 z是x,x是y,y是z,Sample InputSTART 1 ...

  9. 写给Android开发者的混淆使用手册

    转自:http://huihui.name/2016/10/23/%E5%86%99%E7%BB%99Android%E5%BC%80%E5%8F%91%E8%80%85%E7%9A%84%E6%B7 ...

  10. 6-5 移动的盒子 uva12657

    较为复杂的一题:有点类似6-1  但是分析完之后比6-1简单   就是按照思路模拟就好! 学会了双向链表   先初始化   link是关键 分析命令   可以大大简化代码  : 反转链表不用反转  改 ...