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. swift的一些知识点(不断完善中)

    首先,隆重推荐文章http://www.infoq.com/cn/articles/swift-brain-gym-optional swift 烧脑体操!目前有4篇文章,说的都很好! 1. 应该充分 ...

  2. how to detect circles and rectangle?

    opencv中对圆检测的函数为:HoughCircles(src_gray,circles,CV_HOUGHT_GRADIENT,1,src_gray.cols/8,200,100,0,0) circ ...

  3. CentOS7图形界面与命令行界面切换

    1.如果在命令行模式,按Ctrl+Alt+F1,可以进入图形界面: 2.如果在图形界面下,按:Ctrl+Alt+F2,可以进入命令行模式:

  4. .NetCore下利用Jenkins如何将程序自动打包发布到Docker容器中运行

    说道这一块纠结了我两天时间,感觉真的很心累,Jenkins的安装就不多说了 这里我们最好直接安装到宿主机上,应该pull到的jenkins版本是2.6的,里面很多都不支持,我自己试了在容器中安装的情况 ...

  5. django 2中的app_name和namespace

    最近看<精通DJANGO>里遇到的疑惑. 简书URL如下: https://www.jianshu.com/p/404500a0408a 其中的官方解释: 应用命名空间 它表示正在部署的应 ...

  6. [转] webpack热更新配置小结

    webpack热更新配置 热更新,可以使开发的人在修改代码后,不用刷新浏览器即可以看到修改后的效果.而它的另一个好处则是可以只替换修改部分相关的代码,大大的缩短了构建的时间. 热更新一般会涉及到两种场 ...

  7. #10 //I [HNOI/AHOI2018]毒瘤

    题解: 80分做法还是听简单的 对于非树边枚举一下端点状态 然而我也不知道为什么就多t了一个点 具体实现上 最暴力的是3^n次 但是我们可以发现对于i不取,j取 i不取,j不取是可以等效成i不取,j没 ...

  8. 关于final中的几个忽略的点的再次阐述

    final : 最终.作为一个修饰符,可以感性的认识,但是总是在背后会忽略特殊的角落. 1,可以修饰类,函数,变量. 2,被final修饰的类不可以被继承.为了避免被继承,被子类复写功能. 这一个点容 ...

  9. 093实战 Nginx日志切割,以及脚本上传nginx的切割日志

    一:日志切割步骤 命令都在root下进行 1.创建目录 mkdir -p /etc/opt/modules/bin ## 创建文件夹 2.上传cut 3.观察目录 4.修改的cut文件 5.检测 需要 ...

  10. Java内存模型及Java关键字 volatile的作用和使用说明

    先来看看这个关键字是什么意思:volatile  [ˈvɒlətaɪl] adj. 易变的,不稳定的; 从翻译上来看,volatile表示这个关键字是极易发生改变的.volatile是java语言中, ...