C#反射简介
反射能干什么?
使用反射可以让我们在运行时动态地获取对象的类型信息并进行相应的操作,比如创建对象、调用方法、获取属性等。举个简单的例子,我们在写代码时,为了能够调用某个对象的方法,我们通常需要先创建这个对象的实例,然后才能调用其方法。而使用反射机制,我们可以在运行时动态地创建对象并直接调用其方法,而不必提前定义它们。
反射的基本使用
1. 获取类型信息
获取类型信息是反射最基本的用法之一,我们可以使用Type类的静态方法GetType获取类型信息,如下所示。
using System; namespace ReflectionDemo { class Program { static void Main(string[] args { Type type = typeof(string; Console.WriteLine(type.FullName; Console.ReadKey(; } } }
这个例子中,我们获取了string类型的Type对象,然后输出了这个对象的FullName属性,也就是string类型的完全限定名称System.String。
2. 反射创建对象
using System; using System.Reflection; namespace ReflectionDemo { class MyClass { public int Id { get; set; } public string Name { get; set; } } class Program { static void Main(string[] args { // 获取 MyClass 的类型对象 Type myClassType = typeof(MyClass; // 创建 MyClass 类型的实例 MyClass myClass = (MyClassActivator.CreateInstance(myClassType; // 设置对象属性值 PropertyInfo propId = myClassType.GetProperty("Id"; propId.SetValue(myClass, 100; PropertyInfo propName = myClassType.GetProperty("Name"; propName.SetValue(myClass, "Tom"; // 打印对象属性值 Console.WriteLine(myClass.Id; Console.WriteLine(myClass.Name; Console.ReadLine(; } } }
上述代码中,我们首先获取了 MyClass
类型的对象,然后调用 Activator.CreateInstance
方法来创建该类型的实例。接着,我们利用 PropertyInfo
对象获取、设置对象的属性值,最后打印属性值。以上就是用反射机制在 C# 中创建对象的过程。
3. 反射调用方法
using System; using System.Reflection; namespace ReflectionDemo { class Program { static void Main(string[] args { Type type = typeof(string; MethodInfo method = type.GetMethod("ToUpper", new Type[] { }; string result = (stringmethod.Invoke("Hello World", null; Console.WriteLine(result; Console.ReadKey(; } }
}
这个例子中,我们获取了string类型的ToUpper方法信息,然后使用Invoke方法调用这个方法,将字符串"Hello World"转化为大写输出。
反射的高级用法
1. 获取泛型方法信息
使用反射可以在运行时动态地获取泛型方法的信息,然后在运行时构造泛型类型。下面是一个例子。
using System; using System.Reflection; namespace ReflectionDemo { class Program { static void Main(string[] args { Type type = typeof(Program; MethodInfo method = type.GetMethod("TestMethod"; MethodInfo genericMethod = method.MakeGenericMethod(typeof(string; genericMethod.Invoke(null, null; Console.ReadKey(; } public static void TestMethod<T>( { Console.WriteLine(typeof(T.FullName; } } }
这个例子中,我们使用GetMethod方法获取了TestMethod方法信息,然后使用MakeGenericMethod方法构造了泛型方法,并将其转化为MethodInfo类进行输出。
2. 在运行时构造LINQ查询
using System; using System.Collections.Generic; using System.Linq; using System.Linq.Expressions; using System.Reflection; namespace ReflectionDemo { class MyEntity { public int Id { get; set; } public string Name { get; set; } } class Program { static void Main(string[] args { // 构造查询条件 string fieldName = "Id"; int fieldValue = 100; // 获取运行时类型和字段信息 Type entityType = typeof(MyEntity; PropertyInfo property = entityType.GetProperty(fieldName; // 使用表达式树构造查询条件 ParameterExpression parameter = Expression.Parameter(entityType, "x"; MemberExpression member = Expression.Property(parameter, property; ConstantExpression constant = Expression.Constant(fieldValue, property.PropertyType; BinaryExpression equal = Expression.Equal(member, constant; Expression<Func<MyEntity, bool>> expression = Expression.Lambda<Func<MyEntity, bool>>(equal, parameter; // 执行查询 IQueryable<MyEntity> entities = new List<MyEntity> { new MyEntity { Id = 100, Name = "Alice" }, new MyEntity { Id = 200, Name = "Bob" }, new MyEntity { Id = 300, Name = "Charlie" }, new MyEntity { Id = 400, Name = "David" }, }.AsQueryable(; IQueryable<MyEntity> query = entities.Where(expression; // 输出查询结果 foreach (MyEntity entity in query { Console.WriteLine($"Id={entity.Id}, Name={entity.Name}"; } Console.ReadLine(; } static object CreateWhereLambda(Type elementType { MethodInfo whereMethod = typeof(Program.GetMethod(nameof(CreateWhereLambdaImpl, BindingFlags.NonPublic | BindingFlags.Static; return whereMethod.MakeGenericMethod(elementType.Invoke(null, null; } static Func<T, bool> CreateWhereLambdaImpl<T>( { return item => (int(objectitem % 2 == 0; } } }
在上述示例中,我们首先定义了一个查询条件,然后获取了运行时类型和字段信息,接着使用表达式树构造了查询条件,并利用反射执行了 LINQ 查询。最终,我们输出的结果只包括 Id
等于 100 的实体。
反射使用的注意事项
- 尽量使用已经编译好的程序集,避免使用动态编译的程序集。
- 反射的性能较低,尽量少用。
- 反射有漏洞,应注意安全问题。
- 授权可以防止反射的滥用,应根据实际情况授权反射使用权限。
总结
通过本文的学习,我们了解了反射的基本概念和使用方法,并且掌握了反射的高级用法。反射在C#中是一项非常强大且必要的技术,如果恰当地使用它,可以使我们的编程工作变得更加高效和便捷。同时,我们也需要格外注意反射使用过程中的性能和安全问题,做好样本授权等工作,以便更好地使用反射这个强大的功能。