C#反射与特性{学习笔记}

其实这篇文章主要是想要学习反射,但是反射和特性往往是不分家的,所以也要了解一些特性相关的知识.

简单来说,继承了Attribute类的,就是特性
作用是给类或者方法打个标签

反射是在程序运行时,去读取程序集中信息的能力,本质上是解释性的,所以性能不是很强,但是在程序运行中,及时获取新的程序集当中方法时,就可以应用反射

在使用过程中,如果需要使用反射,那么就需要有Type类型的变量

在我们的C#程序经过生成后,其实会被转化成两部分内容:
一部分是IL(也叫CIL)的.NET平台通用汇编语言;
而另一部分则是元数据

C#元数据用于描述程序集、类型、接口、方法、属性和事件等信息

在C#中,元数据可以通过反射来访问。反射是一种允许在运行时检查和操作程序集、类、结构体、枚举、方法和字段等信息的技术。通过反射,可以在运行时获取程序集的元数据信息,例如程序集的名称、版本号和作者等。

获取Type类型变量有两种方式:
一是调用实例化的变量的GetType方法

string a = "";
Type type_a = a.GetType();

二是使用typeof(MyClass)//将在以下demo中展示

Type type = typeof(MyClass);//获得一个类,把它放到Type类型变量里
//要传类名进去哦(不是实例化的对象)

以下是一个反射的极简demo,用来获取类型信息

// See https://aka.ms/new-console-template for more information
using System;
using System.Reflection;
class Program
{
    static void Main(string[] args)
    {
        Type type = typeof(MyClass);//获得一个类,把它放到Type类型变量里
        foreach (PropertyInfo property in type.GetProperties())//遍历他的属性
        {
            Console.WriteLine("Property Name: " + property.Name);
            Console.WriteLine("Property Type: " + property.PropertyType.Name);
            Console.WriteLine();
        }
        foreach (MethodInfo method in type.GetMethods())//遍历方法
        {
            //方法名
            Console.WriteLine("Method Name: " + method.Name);
            //返回值类型
            Console.WriteLine("Method Return Type: " + method.ReturnType.Name);
            //是不是抽象方法
            Console.WriteLine("Is Abstract: " + method.IsAbstract);
            //是不是程序集? 名字是这个意思,但是emm 为什么方法可以是程序集呢
            Console.WriteLine("Is Assembly: " + method.IsAssembly);
            Console.WriteLine("Is Constructor: " + method.IsConstructor);
            Console.WriteLine("Is GenericMethod: " + method.IsGenericMethod);
            Console.WriteLine("Is Private: " + method.IsPrivate);
            Console.WriteLine("Is Public: " + method.IsPublic);
            Console.WriteLine("Is Static: " + method.IsStatic);
            Console.WriteLine();
        }
    }
}
public class MyClass {
    public string Name { get; set; }
    public int Age { get; set; }
    public MyClass() { }
    public void SayHello() { Console.WriteLine("Hello!"); }
}

以下是输出:

这是两个属性:
Property Name: Name
Property Type: String
//
Property Name: Age
Property Type: Int32

这些是方法名:
Method Name: get_Name
Method Return Type: String
Is Abstract: False
Is Assembly: False
Is Constructor: False
Is GenericMethod: False
Is Private: False
Is Public: True
Is Static: False

Method Name: set_Name
Method Return Type: Void
Is Abstract: False
Is Assembly: False
Is Constructor: False
Is GenericMethod: False
Is Private: False
Is Public: True
Is Static: False

Method Name: get_Age
Method Return Type: Int32
Is Abstract: False
Is Assembly: False
Is Constructor: False
Is GenericMethod: False
Is Private: False
Is Public: True
Is Static: False

Method Name: set_Age
Method Return Type: Void
Is Abstract: False
Is Assembly: False
Is Constructor: False
Is GenericMethod: False
Is Private: False
Is Public: True
Is Static: False

Method Name: SayHello
Method Return Type: Void
Is Abstract: False
Is Assembly: False
Is Constructor: False
Is GenericMethod: False
Is Private: False
Is Public: True
Is Static: False

Method Name: GetType
Method Return Type: Type
Is Abstract: False
Is Assembly: False
Is Constructor: False
Is GenericMethod: False
Is Private: False
Is Public: True
Is Static: False

Method Name: ToString
Method Return Type: String
Is Abstract: False
Is Assembly: False
Is Constructor: False
Is GenericMethod: False
Is Private: False
Is Public: True
Is Static: False

Method Name: Equals
Method Return Type: Boolean
Is Abstract: False
Is Assembly: False
Is Constructor: False
Is GenericMethod: False
Is Private: False
Is Public: True
Is Static: False

Method Name: GetHashCode
Method Return Type: Int32
Is Abstract: False
Is Assembly: False
Is Constructor: False
Is GenericMethod: False
Is Private: False
Is Public: True
Is Static: False

这样我们就看到了Type类型能够从MyClass类中读取到的信息了,
而接下来,我们还需要学习:
1.怎样获取程序集

type.assembly;//如以上demo,就可以使用这个属性获取程序集

2.怎样获取具体的方法,并调用

我们需要先获取要调用的方法的MethodInfo对象,然后通过该对象的Invoke方法来调用该方法。
Invoke方法的第一个参数是要调用的方法所在的对象实例,第二个参数是一个包含要传递给方法的参数的数组。

所以,现在已经有了方法,那么,我们只需要把我们所需要的类型的对象实例化出来就可以了,我看到了两种方法
第一种方法

object myClass1 = Activator.CreateInstance(type);//直接实例化这个类
methodInfo?.Invoke(myClass1,null); 

第二种方法错了,文章结束.

以下是附录部分:
第二种方法是我在问星火大模型的时候它告诉我的,但是我没试出来,他说可以获取构造方法然后加以使用(我把代码粘在下面,但是这是错误的代码,它报错,如果大家有会的,可以在评论区留个言啥的,我确实整不明白这个,我觉得有可能是星火乱说的)

//注意这个代码不能用,是报错的仅做参考
using System;
using System.Reflection;
class Program
{
    static void Main()
    {
        // 获取Test类的类型信息
        Type type = typeof(Test);
        // 获取Test类的GetValue方法的MethodInfo对象
        MethodInfo method = type.GetMethod("GetValue");
        // 创建一个Test类的对象实例
        Test test = (Test)method.Invoke(null, new object[] { 1, 2 });
        // 输出结果
        Console.WriteLine(test.GetValue()); // 输出3
    }
}
// Test类的GetValue方法
public class Test
{
    public int GetValue()
    {
        return 1 + 2;
    }
}

posted on 2023-06-16 21:33  嘉君  阅读(29)  评论(1编辑  收藏  举报  来源

导航