反射提供了封装程序集、模块和类型的对象(Type类型)。可以使用反射动态创建类型的实例,将类型绑定到现有对象,或从现有对象获取类型并调用其方法或访问其字段和属性
使用场景
- 需要访问程序元数据的特性
- 检查和实例化程序集中的类型
- 在运行时构建新类型。使用
System.Reflection.Emit中的类 - 执行后期绑定,访问在运行时创建的类型的方法
反射用途
| 类型 | 作用 | | —- | :—- | | Assembly | 定义和加载程序集,加载程序集清单中列出的模块,以及从此程序集中查找类型并创建该类型的实例 | | Module | 了解包含模块的程序集以及模块中的类等,还可以获取在模块上定义的所有全局方法或其他特定的非全局方法 | | ConstructorInfo | 了解构造器的名称、参数、访问修饰符(如public或private)和实现详细信息(如abstract或virtual)等。使用Type的GetConstructors或GetConstructor方法来调用特定的构造函数 | | MethodInfo | 了解方法的名称、返回类型、参数、访问修饰符(如public或private)和实现详细信息(如abstract或virtual)等。使用Type的GetMethods或GetMethod方法来调用特定的方法 | | FieldInfo | 了解字段的名称、访问修饰符(如public或private)和实现详细信息(如static)等,并获取或设置字段值 | | EventInfo | 了解事件的名称、事件处理程序数据类型、自定义特性、声明类型和反射类型等,并添加或移除事件处理程序 | | PropertyInfo | 了解属性的名称、数据类型、声明类型、反射类型和只读或可写状态等,并获取或设置属性值 | | ParameterInfo | 了解参数的名称、数据类型、参数是输入参数还是输出参数等,以及参数在方法签名中的位置等 |
反射用到的命名空间
- System.Reflection
- System.Type
-
反射用到的主要类
System.Type :通过这个类可以访问任何给定数据类型的信息
- System.Reflection.Assembly: 可以用于访问给定程序集的信息,或者把这个程序集加载到程序中
System.Type
一个抽象的基类。Type有与每种数据类型对应的派生类,使用这个派生类的对象的方法、字段、属性来查找有关该类型的所有信息。
获取给定类型Type值
typeof 运算符
Type t = typeof(string);
对象的GetType()方法
string s = "guo";Type t = s.GetType();
Type类的静态方法GetType()
Type t = Type.GetType("System.String");
Type 属性
| 属性 | 描述 |
|---|---|
| Name | 数据类型名 |
| FullName | 数据类型的完全限定名(包括命名空间名) |
| Namespace | 定义数据类型的命名空间名 |
| IsAbstract | 指示该类型是否是抽象类型 |
| IsArray | 指示该类型是否是数组 |
| IsClass | 指示该类型是否是类 |
| IsEnum | 指示该类型是否是枚举 |
| IsInterface | 指示该类型是否是接口 |
| IsPublic | 指示该类型是否是公有的 |
| IsSealed | 指示该类型是否是密封类 |
| IsValueType | 指示该类型是否是值类型 |
Type 方法
| 方法名称 | 描述 |
|---|---|
| GetConstructor(), GetConstructors() | 返回ConstructorInfo类型,用于取得该类的构造函数的信息 |
| GetEvent(), GetEvents() | 返回EventInfo类型,用于取得该类的事件的信息 |
| GetField(), GetFields() | 返回FieldInfo类型,用于取得该类的字段(成员变量)的信息 |
| GetInterface(), GetInterfaces() | 返回InterfaceInfo类型,用于取得该类实现的接口的信息 |
| GetMember(), GetMembers() | 返回MemberInfo类型,用于取得该类的所有成员的信息 |
| GetMethod(), GetMethods() | 返回MethodInfo类型,用于取得该类的方法的信息 |
| GetProperty(), GetProperties() | 返回PropertyInfo类型,用于取得该类的属性的信息 |
可以调用这些成员,其方式是调用 Type 的 InvokeMember() 方法,或者调用 MethodInfo , PropertyInfo 和其他类的 Invoke() 方法。
public class ReflectClass{public string Address;public int Age { get; set; }public string Sex { get; set; }public string Name { get; set; }public ReflectClass() { }public ReflectClass(string name){this.Name = name;}public ReflectClass(string name, string sex){this.Name = name;this.Sex = sex;}public void Show(){Console.WriteLine("姓名:" + Name + "\n" + "年龄:" + Age + "\n" + "性别:" + Sex);}}
获取所有构造函数
[TestMethod]public void GetConstructors(){Type t = new ReflectClass().GetType();// 获取类的所有构造函数ConstructorInfo[] constructorInfos = t.GetConstructors();foreach (ConstructorInfo ci in constructorInfos){// 获取每个构造函数的参数ParameterInfo[] parameterInfos = ci.GetParameters();foreach (ParameterInfo p in parameterInfos){Console.WriteLine(p.ParameterType.ToString() + "\n" + p.Name + "\n");}}Assert.IsTrue(constructorInfos.Length > 0);}
动态创建对象
[TestMethod]public void DynamicCreateObject(){Type t = typeof(ReflectClass);Type[] pt = new Type[2];pt[0] = typeof(string);pt[1] = typeof(string);//根据参数类型获取构造函数ConstructorInfo ci = t.GetConstructor(pt);//构造Object数组,作为构造函数的输入参数object[] obj = new object[2] { "wang", "男" };//调用构造函数生成对象object @object = ci.Invoke(obj);//调用生成的对象的方法测试是否对象生成成功((ReflectClass)@object).Show();Assert.IsTrue(true);}
使用Activator动态创建对象
[TestMethod]public void ActivatorDynamicCreateObject(){Type t = typeof(ReflectClass);object[] obj = new object[2] { "wang", "男" };//用Activator的CreateInstance静态方法,生成新对象object @object = Activator.CreateInstance(t, obj);((ReflectClass)@object).Show();Assert.IsTrue(true);}
获取类中的Public属性
[TestMethod]public void GetProperties(){Type t = new ReflectClass().GetType();PropertyInfo[] propertyInfos = t.GetProperties();foreach (PropertyInfo p in propertyInfos){Console.WriteLine(p.Name);}Assert.IsTrue(true);}
获取类中Public方法
[TestMethod]public void GetPublicMethod(){Type t = new ReflectClass().GetType();MethodInfo[] mi = t.GetMethods();foreach (MethodInfo method in mi){Console.WriteLine(method.ReturnType + "|" + method.Name);}Assert.IsTrue(true);}
获取类中Public字段
[TestMethod]public void GetField(){Type t = new ReflectClass().GetType();FieldInfo[] fieldInfos = t.GetFields();foreach (FieldInfo fieldInfo in fieldInfos){Console.WriteLine(fieldInfo.Name);}Assert.IsTrue(true);}
[TestMethod]public void Example_01(){ReflectClass rc = new ReflectClass();Type t = rc.GetType();object obj = Activator.CreateInstance(t);FieldInfo address = t.GetField("Address");address.SetValue(obj, "Beijing");PropertyInfo name = t.GetProperty("Name");name.SetValue(obj, "wang", null);PropertyInfo age = t.GetProperty("Age");age.SetValue(obj, 20, null);MethodInfo method = t.GetMethod("Show");method.Invoke(obj, null);Console.WriteLine("Address为:" + ((ReflectClass)obj).Address);Assert.IsTrue(true);}
Assembly
GetAssembly
[TestMethod]public void GetAssembly(){// 通过程序集名称返回Assembly对象Assembly assembly = Assembly.Load("CodeSnippet");// 通过Assembly获取程序集中类(参数必须是类的全名)Type type = assembly.GetType("CodeSnippet.Csharp.ReflectClass");Assert.IsNotNull(type);// 通过Assembly获取程序集中所有的类Type[] types = assembly.GetTypes();Assert.IsNotNull(types);// 通过DLL文件名称返回Assembly对象warning disable S3885 // "Assembly.Load" should be usedAssembly assembly2 = Assembly.LoadFrom("CodeSnippet.dll");warning restore S3885 // "Assembly.Load" should be used// 通过Assembly获取程序集中类(参数必须是类的全名)Type type2 = assembly2.GetType("CodeSnippet.Csharp.ReflectClass");Assert.IsNotNull(type2);// 通过Assembly获取程序集中所有的类Type[] types2 = assembly2.GetTypes();Assert.IsNotNull(types2);}
通过程序集的名称反射
[TestMethod]public void Example_02(){Assembly assembly = Assembly.Load("CodeSnippet");//参数必须是类的全名Type t = assembly.GetType("CodeSnippet.Csharp.ReflectClass");object o = Activator.CreateInstance(t, "男");MethodInfo mi = t.GetMethod("Show");mi.Invoke(o, null);Assert.IsNotNull(o);}
通过DLL文件全名反射其中的所有类型
[TestMethod]public void Example_03(){warning disable S3885 // "Assembly.Load" should be usedAssembly assembly = Assembly.LoadFrom("CodeSnippet.dll");warning restore S3885 // "Assembly.Load" should be usedType[] types = assembly.GetTypes();foreach (Type t in types){if (t.FullName == "CodeSnippet.Csharp.ReflectClass"){object o = Activator.CreateInstance(t);Assert.IsNotNull(o);}}}
BindingFlags
// Specifies flags that control binding and the way in which the search for members// and types is conducted by reflection.
