C# 反射

反射是.NET提供的一种方法,它允许在运行时检查类型和调用其成员。在C#中,反射通常用于以下几种情况:

  1. 动态创建类型的实例(无需知道实例的确切类型)。

  2. 在运行时检查类型的成员(方法,属性,字段,事件)。

  3. 调用任何类型的成员,包括方法和属性。

  4. 动态构建复杂的表达式树。

优缺点

优点:

  • 1、反射提高了程序的灵活性和扩展性。
  • 2、降低耦合性,提高自适应能力。
  • 3、它允许程序创建和控制任何类的对象,无需提前硬编码目标类。

缺点:

  • 1、性能问题:使用反射基本上是一种解释操作,用于字段和方法接入时要远慢于直接代码。因此反射机制主要应用在对灵活性和拓展性要求很高的系统框架上,普通程序不建议使用。
  • 2、使用反射会模糊程序内部逻辑;程序员希望在源代码中看到程序的逻辑,反射却绕过了源代码的技术,因而会带来维护的问题,反射代码比相应的直接代码更复杂。

反射获取对象的四种方式

  1. 通过Activator.CreateInstance方法创建对象实例。
  2. 使用Type.InvokeMember方法调用构造函数或其他成员。
  3. 直接使用new关键字创建对象实例(这不是真正的反射,但经常与反射一起使用)。
  4. 使用序列化与反序列化(这也不是直接的反射,但可以用于从数据创建对象)。

以下是每种方式的示例:

1. 使用Activator.CreateInstance

Type type = typeof(MyClass);
object instance = Activator.CreateInstance(type);

2. 使用Type.InvokeMember(包括构造函数)

Type type = typeof(MyClass);
ConstructorInfo constructor = type.GetConstructor(Type.EmptyTypes); // 无参构造函数
object instance = constructor.Invoke(null);

3. 直接使用new关键字

MyClass instance = new MyClass();

4. 使用序列化与反序列化

首先,您需要标记类为可序列化:

[Serializable]
public class MyClass
{
// ...
}

然后,您可以使用例如BinaryFormatterJsonSerializer等序列化器来从数据创建对象:

使用BinaryFormatter(注意:从.NET 5开始不推荐使用,因为它不是跨平台的,并且在某些情况下存在安全风险):

using System.IO;
using System.Runtime.Serialization.Formatters.Binary;

// 序列化
MyClass myObject = new MyClass();
BinaryFormatter formatter = new BinaryFormatter();
using (FileStream stream = new FileStream("myfile.bin", FileMode.Create))
{
formatter.Serialize(stream, myObject);
}

// 反序列化
using (FileStream stream = new FileStream("myfile.bin", FileMode.Open))
{
MyClass deserializedObject = (MyClass)formatter.Deserialize(stream);
}

使用JsonSerializer(推荐用于跨平台和新项目):

using System.IO;
using System.Text.Json;

// 序列化
MyClass myObject = new MyClass();
string jsonString = JsonSerializer.Serialize(myObject);
File.WriteAllText("myfile.json", jsonString);

// 反序列化
string jsonString = File.ReadAllText("myfile.json");
MyClass deserializedObject = JsonSerializer.Deserialize<MyClass>(jsonString);

请注意,使用反射(特别是Activator.CreateInstanceType.InvokeMember)通常比直接使用new关键字创建对象要慢,并且可能会破坏封装性。此外,序列化与反序列化通常用于在不同系统或程序之间传输数据,而不是仅用于在内存中创建对象。在决定使用哪种方法时,请考虑性能、封装性、安全性和易用性等因素。

示例

示例1:使用反射获取和调用方法 

using System;
using System.Reflection;
 
public class MyClass
{
    public void MyMethod()
    {
        Console.WriteLine("MyMethod called");
    }
}
 
class Program
{
    static void Main()
    {
        // 获取类型信息
        Type myType = typeof(MyClass);
 
        // 创建类型的实例
        object myClassInstance = Activator.CreateInstance(myType);
 
        // 获取方法信息
        MethodInfo myMethodInfo = myType.GetMethod("MyMethod");
 
        // 调用方法
        myMethodInfo.Invoke(myClassInstance, null);
    }
}

示例2:使用反射动态创建泛型类型的实例

using System;
using System.Reflection;
 
public class MyGenericClass<T>
{
    public void MyMethod(T value)
    {
        Console.WriteLine($"MyMethod called with value: {value}");
    }
}
 
class Program
{
    static void Main()
    {
        // 获取开放类型
        Type genericType = typeof(MyGenericClass<>);
 
        // 绑定类型参数并创建封闭类型
        Type closedType = genericType.MakeGenericType(typeof(string));
 
        // 创建类型的实例
        object closedClassInstance = Activator.CreateInstance(closedType);
 
        // 获取方法信息
        MethodInfo myMethodInfo = closedType.GetMethod("MyMethod");
 
        // 调用方法
        myMethodInfo.Invoke(closedClassInstance, new object[] { "Hello Reflection!" });
    }
}

示例3:使用反射来获取一个对象(Object)中的字段 

using System;
using System.Reflection;

public class MyClass
{
    public int MyField;
    public string AnotherField;
}

class Program
{
    static void Main()
    {
        // 创建MyClass的实例
        MyClass myObject = new MyClass();
        myObject.MyField = 123;
        myObject.AnotherField = "Hello";

        // 获取MyClass的类型信息
        Type myType = myObject.GetType();

        // 获取所有字段,包括私有和公共字段
        FieldInfo[] fields = myType.GetFields(BindingFlags.Public | BindingFlags.NonPublic | BindingFlags.Instance);

        // 遍历字段并打印信息
        foreach (FieldInfo field in fields)
        {
            Console.WriteLine($"Field Name: {field.Name}, Field Value: {field.GetValue(myObject)}");
        }

        // 如果你知道字段的名称,你也可以直接获取特定的字段
        FieldInfo myField = myType.GetField("MyField", BindingFlags.Public | BindingFlags.NonPublic | BindingFlags.Instance);
        if (myField != null)
        {
            Console.WriteLine($"Specific Field Value: {myField.GetValue(myObject)}");
        }
    }
}

在上面的示例中,我们创建了一个MyClass的实例,并设置了一些字段的值。然后,我们获取了这个实例的类型信息,并使用了GetFields方法来获取所有的字段(包括公共和私有字段)。我们遍历这些字段,并使用GetValue方法来获取每个字段的值。

如果你只想获取特定名称的字段,你可以使用GetField方法,并传入字段的名称和适当的绑定标志。这将返回与该名称匹配的FieldInfo对象,然后你可以使用GetValue来获取它的值。

请注意,如果字段是私有的,你需要使用BindingFlags.NonPublic标志来指示反射应该检索私有成员。如果你只关心公共字段,可以省略这个标志。

反射是一个强大的工具,但它也有一些性能开销,并且可能会破坏封装性,因此应该谨慎使用。在可能的情况下,使用常规的属性、方法等访问成员通常是更好的选择。

  • 7
    点赞
  • 8
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
C# 反射是指在运行时动态地获取类的信息,通过反射可以获取类的属性、方法、事件等信息,以及动态创建对象、调用方法、获取值等操作。这使得我们能够在运行时动态地编写代码,可以写出更加灵活和可扩展的程序。 C# 反射的核心是 `System.Reflection` 命名空间,该命名空间包含了许多与反射相关的类和接口,比如 `Type` 类、`MethodInfo` 类、`PropertyInfo` 类等。 下面是一些常用的反射操作: 1. 获取类型信息 可以使用 `typeof` 关键字或者 `Type.GetType()` 方法来获取类型的信息。`typeof` 关键字用于获取编译时已知的类型信息,而 `Type.GetType()` 方法则可以通过类型名称获取运行时的类型信息。 ```csharp // 获取 System.String 类型的信息 Type type1 = typeof(System.String); // 获取指定类型名称的信息 Type type2 = Type.GetType("System.String"); ``` 2. 获取成员信息 可以使用 `Type.GetMembers()` 方法获取类型的所有成员信息,包括属性、方法、字段、事件等。也可以使用 `Type.GetMethod()`、`Type.GetProperty()`、`Type.GetField()`、`Type.GetEvent()` 等方法获取指定成员的信息。 ```csharp Type type = typeof(Person); // 获取类型的所有成员信息 MemberInfo[] members = type.GetMembers(); // 获取指定属性的信息 PropertyInfo property = type.GetProperty("Name"); // 获取指定方法的信息 MethodInfo method = type.GetMethod("SayHello"); // 获取指定字段的信息 FieldInfo field = type.GetField("Age"); // 获取指定事件的信息 EventInfo eventInfo = type.GetEvent("PropertyChanged"); ``` 3. 动态创建对象 可以使用 `Activator.CreateInstance()` 方法动态创建对象,也可以使用 `Type.InvokeMember()` 方法调用构造函数来创建对象。 ```csharp Type type = typeof(Person); // 使用 Activator.CreateInstance() 方法创建对象 Person person1 = (Person)Activator.CreateInstance(type); // 使用 Type.InvokeMember() 方法调用构造函数创建对象 Person person2 = (Person)type.InvokeMember(null, BindingFlags.CreateInstance, null, null, new object[] { "Tom", 18 }); ``` 4. 调用成员 可以使用 `MethodInfo.Invoke()` 方法调用方法,也可以使用 `PropertyInfo.SetValue()` 方法设置属性的值,使用 `FieldInfo.SetValue()` 方法设置字段的值。 ```csharp Type type = typeof(Person); Person person = new Person("Tom", 18); // 调用方法 MethodInfo method = type.GetMethod("SayHello"); method.Invoke(person, null); // 设置属性的值 PropertyInfo property = type.GetProperty("Name"); property.SetValue(person, "Jerry", null); // 设置字段的值 FieldInfo field = type.GetField("Age"); field.SetValue(person, 20); ``` 以上是 C# 反射的一些基本操作,反射的应用非常广泛,可以用来实现插件式开发、ORM 映射等功能。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值