C#反射基础知识和实战应用

简介: 首先来说一下什么是反射?  反射提供了封装程序集、模块和类型的对象(Type类型) 可以使用反射动态的创建类型的实例,将类型绑定到现有对象,或从现有对象中获取类型,然后,可以调用类型的方法或访问其字段和属性 。

首先来说一下什么是反射? 

反射提供了封装程序集、模块和类型的对象(Type类型)

可以使用反射动态的创建类型的实例,将类型绑定到现有对象,或从现有对象中获取类型,然后,可以调用类型的方法或访问其字段和属性 。

总之,有了反射,以前很多实现不了的功能都可以实现。

下面先来写一个小例子,体验一下反射是怎么一回事:

打开VS2010,新建一个控制台应用程序,在program.cs里面写代码

首先引入命名空间:

using System.Reflection;

下如下代码:

         PropertyInfo len = typeof(string).GetProperty("Length");
         string s = "Hello,reflection!";
         int length = (int)len.GetValue(s, null);
         Console.WriteLine(length.ToString());

这里通过反射获取string的Length属性,并通过调用PropertyInfo 的GetValues方法获取属性值,其中GetValues方法的原型如下:

public virtual object GetValue(object obj, object[] index);

第一个参数obj是拥有此属性的类的实例,在这个例子中,为字符串s,s拥有Length属性。

第二个参数为索引值,微软解释如下:

Optional index values for indexed properties. This value should be null for non-indexed properties.

一般情况下用null,大家可以自己深入研究一下。

GetValues方法返回的是object类型,所以必须强制转换类型。

 

下面通过反射来获取string的一个方法,看看方法是如何通过反射得到的,代码如下:

string s = "Hello,reflection!";
MethodInfo method = typeof(string).GetMethod("Contains");
bool result = (bool)method.Invoke(s, new object[] { "Hello" });
Console.WriteLine(result);

其中,Invoke的方法定义如下:

public object Invoke(object obj, object[] parameters);

这个就很好理解了,第一个参数为拥有此方法的类的实例,还是为string实例s.

第二个参数就是一个object数组的参数。

这里调用的是string中的Contains方法,判断string中是否包含某个字符串。

 

下面通过反射来实例化一个string对象,代码如下:

     Type t = Type.GetType("System.String");
         char[] para = new char[] { 'H', 'e', 'l', 'l', 'o' };
         var o = Activator.CreateInstance(t, para);
         Console.WriteLine(o);

这个跟获取方法相似,唯一不同的就是string的构造方法参数是char[]数组,所以必须传入符合的类型。这里实例化了一个string,值为Hello。

 

看到这里,你对反射已经有了初步的了解,下面开始进入实战应用:

在解决方案上面点击鼠标右键,添加项目,选中类库,输入名称,添加一个类库。

在类库中添加Custom类,代码如下:

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;

namespace ReflectionDll
{
   public class Custom 
   {
      public string Name { get; set; }
      public string Address { get;set; }
      public int Age { get; set; }
      public DateTime BirthDay { get; set; }

      public string GetInfo(string name = "",int age = 0)
      {
         if (name == "" && age == 0)
         {
            return "Custom Name: " + this.Name + ",Age: " + this.Age;
         }
         else
         {
            return "Custom Name: " + name + ",Age: " + age;
         }
      }
   }
}

这里只声明了几个属性和一个方法,供演示使用。写好后编译一下,在控制台项目里面添加引用这个类库(为了方便,否则每次编译都要手动拷贝DLL到控制台项目下面),这样VS会自动将生成的DLL拷贝到控制台debug目录下,方便调用。下面开始使用反射来加载这个DLL,代码如下:

        string path = Environment.CurrentDirectory + "\\ReflectionDll.dll";
         Assembly assem = Assembly.LoadFile(path);
         Type customType = assem.GetType("ReflectionDll.Custom");
         var custom = Activator.CreateInstance(customType);

注意了,这里首先要获取DLL的物理路径,所以上面是否添加引用是无所谓的。有了路径后,通过Assembly的LoadFile方法加载DLL,再获取类的Type,注意GetType方法里面的参数必须为类的全称,及命名空间 + 类名,否则无法找到。

最后一行,创建了一个类的实例,类型为object类型。

下面来获取Custom的所有属性,代码如下:

 PropertyInfo[] propertys = customType.GetProperties();

         Console.WriteLine("******************************");
         foreach (PropertyInfo pro in propertys)
         {
            Console.WriteLine("PropertyName:" + pro.Name + "\n" +
               "PropertyType:" + pro.PropertyType + "\n" +
               "ClassName:" + pro.ReflectedType + "\n" +
               "DLLName:" + pro.Module + "\n");
         }
         Console.WriteLine("******************************");

通过调用GetProperties方法获取所有属性,保存到PropertyInfo[]数组中,然后遍历数组输出属性值。

下面是各个属性的含义:

Name                      属性名称

PropertyType          属性数据类型

ReflectedType         所在的类的命名控件 + 类名

Module                  所在的DLL文件名称

 

设置某个属性的值,方法如下:

     PropertyInfo p = customType.GetProperty("Name");
     p.SetValue(custom, "CustomName",null);

是不是很容易啊。。。

下面来说一下,调用类的方法,和一些属性。代码如下:

         MethodInfo _method = customType.GetMethod("GetInfo");
         //显示方法名称
         Console.WriteLine("Invoke method:" + _method.Name);
         //显示返回的数据类型
         Console.WriteLine("Return type:" + _method.ReturnParameter);
         ParameterInfo[] parameters =  _method.GetParameters();
         foreach (ParameterInfo pa in parameters)
         {
            Console.WriteLine(pa.Name + pa.ParameterType + pa.Position + pa.Member);
         }
         
         object[] paras = new object[] { "Jack",24 };

         Console.WriteLine(_method.Invoke(custom, paras));
同属性一样,参数可以通过GetParameters()来获取,获取的参数信息如下:

Name 参数名称
ParameterType 参数数据类型
Position 参数的位置
Member 输出所有参数

调用有参数的方法时,需要传入参数,New一个object数组,将参数按顺序写入即可。

补充内容:
当类中存在重载方法时,直接获取方法名会报异常,可以通过参数来获取指定的重载方法:
MethodInfo _method = customType.GetMethod("GetInfo",new Type[] {typeof(string),typeof(int)});
 

 



目录
相关文章
|
8天前
|
开发框架 搜索推荐 算法
一个包含了 50+ C#/.NET编程技巧实战练习教程
一个包含了 50+ C#/.NET编程技巧实战练习教程
58 18
|
8天前
|
程序员 C# 数据库
C# 比较对象新思路,利用反射技术打造更灵活的比较工具
中途接手的项目,碰到需要在更新对象信息时比较并记录差异的需求,最变态的还有附加要求,怎么办?有没有既能满足需求又能对项目影响最小的方法呢?分享这个我封装的方法,一个利用反射技术打造的更灵活的比较工具
|
2月前
|
存储 安全 物联网
C# 在物联网 (IoT) 应用中的应用
本文介绍了C#在物联网(IoT)应用中的应用,涵盖基础概念、优势、常见问题及其解决方法。重点讨论了网络通信、数据处理和安全问题,并提供了相应的代码示例,旨在帮助开发者更好地利用C#进行IoT开发。
73 3
|
2月前
|
编译器 C#
c# - 运算符<<不能应用于long和long类型的操作数
在C#中,左移运算符的第二个操作数必须是 `int`类型,因此需要将 `long`类型的位移计数显式转换为 `int`类型。这种转换需要注意数据丢失和负值处理的问题。通过本文的详细说明和示例代码,相信可以帮助你在实际开发中正确使用左移运算符。
40 3
|
2月前
|
编译器 C#
c# - 运算符<<不能应用于long和long类型的操作数
在C#中,左移运算符的第二个操作数必须是 `int`类型,因此需要将 `long`类型的位移计数显式转换为 `int`类型。这种转换需要注意数据丢失和负值处理的问题。通过本文的详细说明和示例代码,相信可以帮助你在实际开发中正确使用左移运算符。
67 1
|
3月前
|
开发框架 NoSQL MongoDB
C#/.NET/.NET Core开发实战教程集合
C#/.NET/.NET Core开发实战教程集合
|
2月前
|
编译器 C#
c# - 运算符<<不能应用于long和long类型的操作数
在C#中,左移运算符的第二个操作数必须是 `int`类型,因此需要将 `long`类型的位移计数显式转换为 `int`类型。这种转换需要注意数据丢失和负值处理的问题。通过本文的详细说明和示例代码,相信可以帮助你在实际开发中正确使用左移运算符。
21 0
|
4月前
|
设计模式 开发框架 前端开发
MVC 模式在 C# 中的应用
MVC(Model-View-Controller)模式是广泛应用于Web应用程序开发的设计模式,将应用分为模型(存储数据及逻辑)、视图(展示数据给用户)和控制器(处理用户输入并控制模型与视图交互)三部分,有助于管理复杂应用并提高代码可读性和维护性。在C#中,ASP.NET MVC框架常用于构建基于MVC模式的Web应用,通过定义模型、控制器和视图,实现结构清晰且易维护的应用程序。
69 2
|
3月前
|
消息中间件 网络协议 安全
C# 一分钟浅谈:WebSocket 协议应用
【10月更文挑战第6天】在过去的一年中,我参与了一个基于 WebSocket 的实时通信系统项目,该项目不仅提升了工作效率,还改善了用户体验。本文将分享在 C# 中应用 WebSocket 协议的经验和心得,包括基础概念、C# 实现示例、常见问题及解决方案等内容,希望能为广大开发者提供参考。
189 0
|
3月前
|
Web App开发 网络协议 API
基于C#编写一个远程桌面应用
基于C#编写一个远程桌面应用
78 0