1、了解什么事程序集

2、加载程序集

首先要加载需要加载的程序集,然后找到指定的类型,进而往下进行动态加载。

要加载的程序集中的内容:

     public class Class1:Person
{
private string name;
private int age;
private void SayHi()
{
Console.WriteLine("hello shit!");
}
} public delegate void VoidDel(); public class MyClass
{
private void MySayHi()
{
Console.WriteLine("my hello shit!");
}
}
internal enum Hopppy
{
游泳,
篮球,
足球
} public class Person
{
public string Name { set; get; }
public int Age { set; get; } public void Shit()
{
Console.WriteLine("I love you,shit!");
}
} public class Chinses : Person, IXiFuable
{
public string Hoppy { set; get; } public void XiFu()
{
Console.WriteLine("中国人又自我修复的功能!");
}
} interface IXiFuable
{
void XiFu();
} public abstract class MyAbsClass
{
}

程序集中的内容

加载程序集:

             //反射:命名空间(System.Reflection;);用于指向特定的程序集,返回程序集的元数据

             //加载程序集(Assembly:程序集)
Assembly asm = Assembly.LoadFile(@"C:\练习\委托和多线程\基础练习\clRefletor\bin\Debug\clRefletor.dll");

获得程序集下面的所有类型(包括私有的和internal)

 Type[] types = asm.GetTypes();
foreach (var item in types)
{
//(包括私有和internal)类型名称 命名空间 命名空间.类型名称
Console.WriteLine("Name:" + item.Name + ";namespace:" + item.Namespace + ";FullName:" + item.FullName);
}

获得程序集下面所有的“公有”的类型

Type[] ExportedTypes= asm.GetExportedTypes();

3、获得指定类型的三种方法(反射的核心)

含有一个测试的Person类:

     public class Person
{
public string Name { set; get; }
public int Age { set; get; }
}

获得Type的三种方法

             //1、通过 实例对象的GetType()方法
Person person = new Person();
Type type1 = person.GetType(); //2、通过 typeof(对象的类型名(例如:Person(就是类Person)))
Type type2 = typeof(Person); //3、通过 程序集Assembly
Assembly asm = Assembly.LoadFile("扩展程序集的地址");
Type type3 = asm.GetType("Person");

4、Type类型的动态调用成员

得到类型之后,就能找到他下面的所有的属性、字段、方法、接口、事件等

获得Type下面的所有的”公有“属性:

            Type asmType = asm.GetType("clRefletor.Class1", true, false);//第一个是“类型的全名”,第二个参数:找不到时触发异常,第三个参数:寻找的时候是否忽略大小写

            PropertyInfo[] propInfos = asmType.GetProperties();
foreach (var item in propInfos)
{
Console.WriteLine("属性Name:" + item.Name + "属性的类型名称:" + item.PropertyType.Name + ";是否可读:" + item.CanRead + ";是否可写:" + item.CanWrite);
}

获得Type下面的所有的“公有”方法:

             //这个时候 会将所有的方法全部都 取出来,包括 继承父类的方法和  属性执行的两个方法(非private和非protected的方法)
MethodInfo[] asmMenthods = asmType.GetMethods();
foreach (var asmMethod in asmMenthods)
{
Console.WriteLine(asmMethod.Name + ";声明此方法的类:" + asmMethod.DeclaringType.Name + asmMethod);
}

上面 动态获得的都是“公有”,要想获得私有的要进行设置

获得私有的方法:

             Type typePerson=typeof(Person);
//获得非私有的 方法( BindingFlags.NonPublic | BindingFlags.Instance主要靠这个枚举)
MethodInfo perMethod= typePerson.GetMethod("SayHi", BindingFlags.NonPublic | BindingFlags.Instance);
object obj = Activator.CreateInstance(typePerson);
perMethod.Invoke(obj, null);//方法没有参数,所以传值 为null
//获得所有的 非公有的方法
MethodInfo[] perMothods = typePerson.GetMethods(BindingFlags.NonPublic|BindingFlags.Instance);

动态获得其他内容就不在演示,基本都一样。

 5、通过Type创建对应的实例对象,进而通过反射设置其属性的值、取到属性的值、调用对应的方法等

通过Type类型创建实例对象

             //获得当前类的实例,,就可以 赋值 此类的属性 和取得属性的值,还可以执行此类中的 方法(都是共有的,因为私有的取不到)
object classTarget = Activator.CreateInstance(asmType);

通过实例对象对属性赋值

             //为属性赋值
var pror = asmType.GetProperty("Name");
pror.SetValue(classTarget, "shit",null);//第三个参数 只有索引才填
Console.WriteLine(pror.GetValue(classTarget,null));

通过实例对象调用指定的方法

             //执行 通过反射得到的方法
MethodInfo method = asmType.GetMethod("Shit");
method.Invoke(classTarget, null);//第一个参数:如果方法是静态方法,就传递一个null就可以了,如果是实例方法,就是调用此方法的 对象(这里是Class1);第二个参数:object数组:传到方法里面的 参数(参数的个数不同,可以识别 方法的重载)

 6、Type类型常用的验证方法和需要注意点

首先获得下面需要用到的类型

             Assembly asm = Assembly.LoadFile(@"C:\练习\委托和多线程\基础练习\clRefletor\bin\Debug\clRefletor.dll");
Type typePerson = asm.GetType("clRefletor.Person");
Type typeChinese = asm.GetType("clRefletor.Chinses");
Type typeIXiufuable = asm.GetType("clRefletor.IXiFuable");
Type typeAbsClass = asm.GetType("clRefletor.MyAbsClass");

1》IsAssignableFrom (Type)

判断方法里的类型是否可以赋值给当前类型。(注意:接口和父类都可以

             bool b1 = typePerson.IsAssignableFrom(typeChinese);//True
bool b2 = typeIXiufuable.IsAssignableFrom(typeChinese);//True

2》IsInstanceOfType(实例对象)

判断括号中的实例对象是否当前类型的实例。(注意:父类类型、接口类型、当前类型都可以

             //动态创建对象(含有无参构造函数)
object objChinese = Activator.CreateInstance(typeChinese); bool b1 = typeChinese.IsInstanceOfType(objChinese);//True
bool b2 = typePerson.IsInstanceOfType(objChinese);//True
bool b3 = typeIXiufuable.IsInstanceOfType(objChinese);//True

3》IsSubclassOf(Type)

判断当前类型是否是括号中类型的子类。(注意:父类可以,但是接口不行

             bool b1 = typeChinese.IsSubclassOf(typePerson);//True
bool b2 = typeChinese.IsSubclassOf(typeIXiufuable);//False

4》IsAbstract属性

判断当前类型是否是抽象的。(注意:抽象的:是指只要不能实例化就是,比如:静态类、密封类、接口等

             Console.WriteLine(typeChinese.IsAbstract);//True
Console.WriteLine(typePerson.IsAbstract);//True
Console.WriteLine(typeIXiufuable.IsAbstract);//False
Console.WriteLine(typeAbsClass.IsAbstract);//False

c#之反射总结的更多相关文章

  1. 隐私泄露杀手锏 —— Flash 权限反射

    [简版:http://weibo.com/p/1001603881940380956046] 前言 一直以为该风险早已被重视,但最近无意中发现,仍有不少网站存在该缺陷,其中不乏一些常用的邮箱.社交网站 ...

  2. Java学习之反射机制及应用场景

    前言: 最近公司正在进行业务组件化进程,其中的路由实现用到了Java的反射机制,既然用到了就想着好好学习总结一下,其实无论是之前的EventBus 2.x版本还是Retrofit.早期的View注解框 ...

  3. 关于 CSS 反射倒影的研究思考

    原文地址:https://css-tricks.com/state-css-reflections 译者:nzbin 友情提示:由于演示 demo 的兼容性,推荐火狐浏览.该文章篇幅较长,内容庞杂,有 ...

  4. 编写高质量代码:改善Java程序的151个建议(第7章:泛型和反射___建议106~109)

    建议106:动态代理可以使代理模式更加灵活 Java的反射框架提供了动态代理(Dynamic Proxy)机制,允许在运行期对目标类生成代理,避免重复开发.我们知道一个静态代理是通过主题角色(Prox ...

  5. 运用Mono.Cecil 反射读取.NET程序集元数据

    CLR自带的反射机智和API可以很轻松的读取.NET程序集信息,但是不能对程序集进行修改.CLR提供的是只读的API,但是开源项目Mono.Cecil不仅仅可以读取.NET程序集的元数据,还可以进行修 ...

  6. .NET面试题系列[6] - 反射

    反射 - 定义,实例与优化 在面试中,通常会考察反射的定义(操作元数据),可以用反射做什么(获得程序集及其各个部件),反射有什么使用场景(ORM,序列化,反序列化,值类型比较等).如果答得好,还可能会 ...

  7. .NET基础拾遗(4)委托、事件、反射与特性

    Index : (1)类型语法.内存管理和垃圾回收基础 (2)面向对象的实现和异常的处理基础 (3)字符串.集合与流 (4)委托.事件.反射与特性 (5)多线程开发基础 (6)ADO.NET与数据库开 ...

  8. C++的性能C#的产能?! - .Net Native 系列五:.Net Native与反射

    此系列系小九的学堂原创翻译,翻译自微软官方开发向导,一共分为六个主题.本文是第五个主题:.Net Native与反射. 向导文链接:<C++的性能C#的产能?! - .Net Native 系列 ...

  9. [源码]Literacy 快速反射读写对象属性,字段

    Literacy 说明 Literacy使用IL指令生成方法委托,性能方面,在调用次数达到一定量的时候比反射高很多 当然,用IL指令生成一个方法也是有时间消耗的,所以在只使用一次或少数几次的情况,不但 ...

  10. SI与EMI(一) - 反射是怎样影响EMI

    Mark为期两天的EMC培训中大概分成四个时间差不多的部分,简单来说分别是SI.PI.回流.屏蔽.而在信号完整性的书籍中,也会把信号完整性分为:1.信号自身传输的问题(反射,损耗):2.信号与信号之间 ...

随机推荐

  1. java基础-在dos控制台编写简易的java程序

    第一步:在文件夹中修改隐藏的文件扩展名,让其文件的扩展名全部显示: 第二步:在文件夹中新建一个text文件,将其扩展名属性改为Hello.java的文件扩展名: 第三步:点击右键打开方式用txt文本打 ...

  2. C#之使用随机数

    1.C#自带随机数函数 using System; System.Random ran = new System.Random();int n = ran.Next(100, 1000);//产生10 ...

  3. php学习笔记6--php中的文件包含 include,require,include_once,require_once

    php中的文件包含 include,require,include_once,require_once 文件包含:是指将一个文件的内容包含进另外一个文件,有利于代码的复用等.php中文件包含指令有4个 ...

  4. Hive通过查询语句向表中插入数据注意事项

    最近在学习使用Hive(版本0.13.1)的过程中,发现了一些坑,它们或许是Hive提倡的比关系数据库更加自由的体现(同时引来一些问题),或许是一些bug.总而言之,这些都需要使用Hive的开发人员额 ...

  5. jQuery中的DOM操作<思维导图>

    DOM是Document Object Model的缩写,意思是文档对象模型.DOM是一种与浏览器.平台.语言无关的接口.使用该接口可以轻松地访问页面中所有的标准组件.简单来说,DOM解决了Netsc ...

  6. [SQL]SUTFF内置函数的用法 (删除指定长度的字符并在指定的起始点插入另一组字符)

    STUFF 删除指定长度的字符并在指定的起始点插入另一组字符. 语法 STUFF ( character_expression , start , length , character_express ...

  7. CSS里的引用@import、link

    引入CSS的方法有两种,一种是@import,一种是link @import url('地址');<link href="地址" rel="stylesheet&q ...

  8. [老老实实学WCF] 第一篇 Hello WCF

    老老实实学WCF  第一篇 Hello WCF WCF(Windows Communication Foundation)是微软公司推出的面向服务技术的集大成者,涵盖继承了其之前发布的所有的分布式应用 ...

  9. 构建前端Mock Server

    写在前面 最开始只是在做活动页面时苦于效率太低制定了这样一个自动化的工作环境, 所以Github上项目名是Rapid-Dev-Activity-Page(快速开发活动页...). 活动页这类比较简单的 ...

  10. spring事务配置的坑

    基于 <tx> 命名空间的声明式事务管理 前面两种声明式事务配置方式奠定了 Spring 声明式事务管理的基石.在此基础上,Spring 2.x 引入了 <tx> 命名空间,结 ...