【点滴积累】在C#中使用反射动态调用类型成员

简介:

1.背景

  学习C#的时候就知道使用反射可以对我们编程提供极大的便利(动态的获取信息、调用类型成员、创建实例等等),但是在实际的工作中直接使用反射的机会比较少,有印象的就是一次自己做的WinForms小工具的时候利用反射来动态获取窗体上的每个控件,并且为必要的控件动态添加注册事件。因为刚入职新公司,为了更快的了解公司的业务、和开发习惯,先和现在公司同事一起修改现有系统的一些小Bug。在QA提交的Bug中有一个是对GridView进行动态的排序——点击一个列时使用该列作为条件进行排序(PS:点击一个列时前台会将该列的字符串(该字符串是)传到后台的方法中)。

 

2.使用反射的原因

  为什么会选择使用反射呢?在项目中我们使用NHibernate作为ORM框架,一般情况下在BLL层其实都提供了对应的排序方法,但是现在需要修改的这个Bug使用的数据不是在一张表里面,而是一个视图。如果按照以前排序功能实现模式的话,需要在多个地方进行修改;且由于是刚接触这个项目,对项目的结构还不够了解,不宜进行大面积的修改。和同事沟通后决定使用反射来实现。理由如下:

  • 因为是直接对已经从数据库返回的结果进行排序,所有只需要在一个地方修改(BLL层)
  • 虽然使用反射会增加性能上的开销,但是在这个场景数据量并不大,性能上的消耗可以忽略不计

 

3.使用反射动态调用代码

  反射给人感觉好像很复杂,其实使用起来还是比较方便的,由于项目中代码上下文牵涉过多,不适合贴出来,下面是使用我自己整理的使用反射进行动态调用,常用的一些代码片段:

复制代码
 1     class Program
 2     {
 3         static void Main(string[] args)
 4         {
 5             Type type = typeof(Employee);
 6 
 7             //使用无参构造函数动态创建对象
 8             var objNull = type.InvokeMember(null, BindingFlags.CreateInstance, null, null, null);
 9 
10             //调用两个使用了两个string参数的构造函数动态创建对象
11             var frankJob = type.InvokeMember(null, BindingFlags.CreateInstance, null, null, new object[] { "job", "frank" });
12 
13             //调用公有成员属性get方法
14             var fileName = type.InvokeMember("FirstName", BindingFlags.GetProperty, null, frankJob, null);
15 
16             //调用公有成员属性set方法
17             type.InvokeMember("Email", BindingFlags.SetProperty, null, frankJob, new object[] { "gyzhao@vervidian.com" });
18 
19             //动态调用无参数方法
20             var objStr = type.InvokeMember("ToString", BindingFlags.InvokeMethod | BindingFlags.Public | BindingFlags.Instance | BindingFlags.Static, null, frankJob, null);
21 
22             //动态调用带参数的方法
23             var email = type.InvokeMember("GetEmail", BindingFlags.InvokeMethod, null, frankJob, new object[] { "sunshine" });
24         }
25 
26         public class Employee
27         {
28             public int Id { get; set; }
29             public string FirstName { get; set; }
30             public string LastName { get; set; }
31             public string Address { get; set; }
32             public string Email { get; set; }
33 
34             public Employee() { }
35             public Employee(string firstName, string lastName)
36             {
37                 FirstName = firstName;
38                 LastName = lastName;
39             }
40 
41             public override string ToString()
42             {
43                 return string.Format("{0} {1}", LastName, FirstName);
44             }
45 
46             public string GetEmail(string user)
47             {
48                 return string.Format("{0}@gmail.com", user);
49             }
50         }
51     }
复制代码

 

参考资料&进一步阅读

.NET Framework 中的反射

 

本文转自gyzhao博客园博客,原文链接:http://www.cnblogs.com/IPrograming/archive/2013/05/25/reflection_dynamic_Invoke.html ,如需转载请自行联系原作者
相关文章
|
2月前
|
C#
一文搞懂C#中类成员的可访问性
一文搞懂C#中类成员的可访问性
39 5
|
10天前
|
C#
C# 可空类型(Nullable)
C# 单问号 ? 与 双问号 ??
36 12
|
3天前
|
C#
C# 可空类型(Nullable)
C# 提供了一个特殊的数据类型,nullable 类型(可空类型),可空类型可以表示其基础值类型正常范围内的值,再加上一个 null 值。
|
2月前
|
存储 C#
揭秘C#.Net编程秘宝:结构体类型Struct,让你的数据结构秒变高效战斗机,编程界的新星就是你!
【8月更文挑战第4天】在C#编程中,结构体(`struct`)是一种整合多种数据类型的复合数据类型。与类不同,结构体是值类型,意味着数据被直接复制而非引用。这使其适合表示小型、固定的数据结构如点坐标。结构体默认私有成员且不可变,除非明确指定。通过`struct`关键字定义,可以包含字段、构造函数及方法。例如,定义一个表示二维点的结构体,并实现计算距离原点的方法。使用时如同普通类型,可通过实例化并调用其成员。设计时推荐保持结构体不可变以避免副作用,并注意装箱拆箱可能导致的性能影响。掌握结构体有助于构建高效的应用程序。
65 7
|
2月前
|
程序员 C#
C# 语言类型全解
C# 语言类型全解
15 0
|
2月前
|
开发框架 .NET 编译器
C# 中的记录(record)类型和类(class)类型对比总结
C# 中的记录(record)类型和类(class)类型对比总结
|
2月前
|
传感器 开发框架 JSON
聊聊 C# dynamic 类型,并分享一个将 dynamic 类型变量转为其它类型的技巧和实例
聊聊 C# dynamic 类型,并分享一个将 dynamic 类型变量转为其它类型的技巧和实例
|
4月前
|
开发框架 .NET 编译器
程序与技术分享:C#基础知识梳理系列三:C#类成员:常量、字段、属性
程序与技术分享:C#基础知识梳理系列三:C#类成员:常量、字段、属性
30 2
|
4月前
|
机器学习/深度学习 开发框架 .NET
技术经验解读:【c#】反射真的很可怕吗?
技术经验解读:【c#】反射真的很可怕吗?
|
5月前
|
编译器 C#
C#.Net筑基-类型系统②常见类型 --record是什么类型?
`record`在C#中是一种创建简单、只读数据结构的方式,常用于轻量级数据传输。它本质上是类(默认)或结构体的快捷形式,包含自动生成的属性、`Equals`、`ToString`、解构赋值等方法。记录类型可以继承其他record或接口,但不继承普通类。支持使用`with`语句创建副本。例如,`public record User(string Name, int Age)`会被编译为包含属性、相等比较和`ToString()`等方法的类。记录类型提供了解构赋值和自定义实现,如密封的`sealed`记录,防止子类重写。