C#高阶-反射

Posted

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了C#高阶-反射相关的知识,希望对你有一定的参考价值。

参考技术A 概念

反射机制是使程序具有动态特性的非常关键的一种机制

需求1:根据指定的类名,类字段名和所对应的数据,得到该类的实例。

需求2:根据指定的装箱对象 获取其属性值或者调用其方法

需求3:获取未来会产生的程序集 并使用其中的类

让我们可以于运行时加载、探知、使用编译期间完全未知的classes。换句话说,程序可以加载一个运行时才得知名称的class,获悉其完整构造(但不包括methods定义),并生成其对象实体、或对其fields设值、或唤起其methods1。

反射需要的类

反射用到的命名空间:

   System.Reflection

   System.Type

   System.Reflection.Assembly

   (1)使用Assembly定义和加载程序集,加载在程序集清单中列出模块,以及从此程序集中查找类型并创建该类型的实例。    

   (2)使用Type这个类可以访问任何给定数据类型的信息。

   (3)使用ConstructorInfo了解构造函数的名称、参数、访问修饰符(如pulic 或private)和实现详细信息(如abstract或virtual)等。

   (4)使用MethodInfo了解方法的名称、返回类型、参数、访问修饰符(如pulic 或private)和实现详细信息(如abstract或virtual)等。

   (5)使用FiedInfo了解字段的名称、访问修饰符(如public或private)和实现详细信息(如static)等,并获取或设置字段值。

   (6)使用PropertyInfo了解属性的名称、数据类型、声明类型、反射类型和只读或可写状态等,获取或设置属性值。

System.Type类:

   System.Type 类对于反射起着核心的作用。但它是一个抽象的基类,Type有与每种数据类型对应的派生类,我们使用这个派生类的对象的方法、字段、属性来查找有关该类型的所有信息。

   获取给定类型的Type引用有3种常用方式:

   ●使用 C# typeof 运算符。已知类型

       Type t = typeof(string);

   ●使用对象GetType()方法。 已知对象

       string s = "grayworm";

       Type t = s.GetType();

   ●还可以调用Type类的静态方法GetType()。已知类名

       Type t = Type.GetType("System.String");

反射生成的对象

1.用Type构建对象

用构造函数动态生成对象:

既然有反射 那就有“正射”

对比一下:

有一个class Test  有无参构造方法

获取实例的比较:

“正” :由类名直接使用关键字new来获得 语法:new Test()

反射:未知类名 运行时获得类型名:例如:Type type = Type.GetType("Test");

   Object o = type.GetConstructor (new Type[]).Invoke (null);

  当然参数也可以由配置文件获得

以上调用无参构造 带参的构造怎么调用呢?

2.用Activator生成对象:

      Type t = typeof(NewClassw);

       //构造函数的参数

       object[] obj = new object[2] "实参1", "实参2" ;   

       //用Activator的CreateInstance静态方法,生成新对象

       object o = Activator.CreateInstance(t,obj);

3.Assembly生成对象:

Assembly assembly = Assembly.Load ("ReflectTest");

Object obj = assembly.CreateInstance ("ReflectTest.Wood");

反射获取字段值

一般用于已知对象时使用

同样用正反来做对比:

已知对象MyClass obj

对象内有name这个字段

正:

obj.name = "xxx"即可设置该对象的值

obj.name即可获得该对象的值

反射:

//获取类的描述对象

Type t = obj.GetType();

//取得字段描述对象

FieldInfo fi = t.GetField("name");

//获取或设置字段值

fi.SetValue(obj, "k001");

fi.GetValue(obj);

一般在C#中以属性替代字段

//获取类的描述对象

Type t = obj.GetType();

//取得属性描述对象

PropertyInfo pi1 = t.GetProperty("Name");

//获取或设置属性值

pi1.SetValue(obj, "k001",null);

pi1.GetValue(obj,null);

反射执行方法

一般用于对象、方法名、参数类型已知

对象:obj 

方法名:methodName

参数类型 String int

正:

直接给予参数调用:obj.methodName("实参1",实参2(int))

反射:

//获取类的描述对象

Type t = obj.GetType();

//取得方法描述对象

MethodInfo mi = t.GetMethod("show");

//调用方法

mi.Invoke(obj, "实参1",实参2(int));

参数可以换成数组

动态载入程序集

System.Reflection.Assembly类

    Assembly类可以获得程序集的信息,也可以动态的加载程序集,以及在程序集中查找类型信息,并创建该类型的实例。

   使用Assembly类可以降低程序集之间的耦合,有利于软件结构的合理化。

   通过程序集名称返回Assembly对象

       Assembly ass = Assembly.Load("ClassLibrary831");

   通过DLL文件名称返回Assembly对象

       Assembly ass = Assembly.LoadFrom("ClassLibrary831.dll");

   通过Assembly获取程序集中类

       Type t = ass.GetType("ClassLibrary831.NewClass");   //参数必须是类的全名

   通过Assembly获取程序集中所有的类

       Type[] t = ass.GetTypes();

   //通过程序集的名称反射

   Assembly ass = Assembly.Load("ClassLibrary831");

   Type t = ass.GetType("ClassLibrary831.NewClass");

   object o = Activator.CreateInstance(t, "", "");

   MethodInfo mi = t.GetMethod("show");

   mi.Invoke(o, null);

  //通过DLL文件全名反射其中的所有类型

   Assembly assembly = Assembly.LoadFrom("xxx.dll的路径");

   Type[] aa = a.GetTypes();

   foreach(Type t in aa)

   

       if(t.FullName == "a.b.c")

       

           object o = Activator.CreateInstance(t);

       

   

小结

反射属于面向对象的动态补充

让C#的面向对象做到"所见即所得"

反射,面对对象高阶

1\  反射

面对对象通常访问属性,是通过  对象.数据属性访问的   可以得到类的,对象的值

class People:
    school=oldboy
    def __init__(self,name):
        self.name=name

    def tell_info(self):
        print(%s is telling%self.name)

p=People(egon)
print(People.school)#oldboy
print(p.name) #egon


现在通过字符串可以得到属性的值

print(hasattr(People,‘school‘))#True  现在可以通过hasattr函数用字符串‘school‘判断能够得到类的值

 print(getattr(People,‘school‘))#oldboy 通过getattr  通过字符串获取值

print(getattr(People,‘sch‘,default=‘no‘))  #,没有‘sch‘,没有找到关键字 参数,就报错
setattr(People,‘x‘,123) #通过字符串形式设置值
print(People.x) #得到值
delattr(People,‘school‘)#通过字符串删除属性和值

hasattr ,setattr,delattr 都是通过类的属性用字符串来操作值,


#==========对象,通过字符串得到的值

p=People(‘egon‘,‘mail‘)
print(hasattr(p,‘name‘))#True 对象通过字符串得到判断
print(getattr(p,‘name‘))#egon
setattr(p,‘age‘,11) #给对象设置新的属性,age,值为11
print(getattr(p,‘age‘))#11 getattr 通过字符串得到值

setattr(p,‘age‘,11)  #给对象设置新的属性,age,值为11
print(getattr(p,‘age‘))#11 getattr 通过字符串得到值

delattr(p,‘sex‘)   #通过字符串删除对象的属性
print(p.__dict__)#‘name‘: ‘egon‘, ‘age‘: 11}

p.tell_info()
print(p.school)
print(getattr(p,‘tell_info‘))#拿到了绑定方法<bound method People.tell_info of <__main__.People object at 0x0000016D79559D68>>
print(getattr(p,‘school‘))
p.school=‘ddd‘ #修改了自己的数据属性
print(p.__dict__)#{‘name‘: ‘egon‘, ‘sex‘: ‘mail‘, ‘school‘: ‘ddd‘}
print(People.__dict__)#{‘__module__‘: ‘__main__‘, ‘school‘: ‘oldboy‘, ‘_类里的没有改动

#===以上都是通过字符串反射得到属性真实的值, 用在与用户的交互的场景


2\ 改变对象的字符串显示__str__,__repr__
#自己定义的类在打印时,得到内存地址,
#内置的__str__的方法

class People:
    school=oldboy
    def __init__(self,name,sex):
        self.name=name
        self.sex=sex
    def tell_info(self):
        print(%s is telling%self.name)
    def __str__(self):  #一定要返回值,必须是字符串,再打印对象 p 时,就是返回的值了
       # return aaa
        return name %s is sex %s%(self.name,self.sex)

p=People(egon,mail)
print(p)  #p.__str__()
#name egon is sex mail

#不写__str__,打印对象p ,得到的是内存地址,由__str__打印对象p 就得到了对象的值
#直接打印对象p ,得到内存地址,<__main__.People object at 0x00000270DAB99D68>,触发了__str__方法,
#TypeError: __str__ returned non-string (type NoneType)
#报错,返回了不是字符串类型的

# print(p.name)

 





以上是关于C#高阶-反射的主要内容,如果未能解决你的问题,请参考以下文章

反射,面对对象高阶

简谈Java 反射机制,动态代理

简谈Java 反射机制,动态代理

C#反射与特性:反射基础

C# 反射

c# 如何通过反射 获取属性值