基础命名空间:反射 using System.Reflection
反射概念:
.Net的应用程序由几个部分:‘程序集(Assembly)’、‘模块(Module)’、‘类型(class)’组成,程序集包含模块 模块包含类型,类型又包含
成员,而反射提供一种编程的方式,让程序员可以在程序运行期获得这几个组成部分的相关信息, .NET Framework提供命名空间System.Refection和
System.Type,可以获取已加载的程序集和程序集中定义的类型(类、接口、值类型)的信息也可以使用反射在运行时创建类型实例,将类型绑定到现有
对象,或主从现有对象中获取类型,调用和访问这些实例
练习示例:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
using Business; namespace FirstReflection
{
public class ComputerFactory : IMessageHandler
{
//品牌
public string brand { get; set; }
//地址
public string address { get; set; } /// <summary>
/// 构造函数
/// </summary>
public ComputerFactory()
{ }
/// <summary>
/// 构造函数
/// </summary>
/// <param name="brand"></param>
public ComputerFactory(string brand,string address)
{
this.brand = brand;
this.address = address;
} public void displayBrand()
{
Console.WriteLine(this.brand);
} public void displayaddress(string address)
{
Console.WriteLine("public:生产地址:{0}",address);
} public static void displayaddress()
{
Console.WriteLine("public static共有方法:生产地址:本部");
} private void privatedisplayaddress(string address)
{
Console.WriteLine("private私有方法:生产地址:{0}", address);
}
}
}
通过Assembly类的load()方法加载程序集,并在程序集中查找定义的类型......
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
using System.Reflection; namespace FirstReflection
{
/*反射:
1.获取方法的信息
2.调用方法
3.构造对象
4.从程序集中加载类型
*/ public class Program
{
//查看程序集中所有的类
public static void ShowClassName(Type[] types)
{
Console.WriteLine("查看程序集中所有的类:");
foreach (Type type in types)
{
Console.WriteLine(type);
}
} public static void ShowModuleName(Module[] modules)
{
Console.WriteLine("查看程序集中所有的模块:");
foreach (Module module in modules)
{
Console.WriteLine(module);
}
} public static void ShowPropertyName(PropertyInfo[] properties)
{
Console.WriteLine("查看该类型的属性:");
foreach (PropertyInfo property in properties)
{
Console.WriteLine(property);
}
} static void Main(string[] args)
{ //[0]
//加载程序集
Assembly assembly = Assembly.Load("FirstReflection");
//Assembly assembly = Assembly.LoadFrom("FirstReflection.exe"); //[1]:查看程序集中所有的类
Type[] types = assembly.GetTypes();
ShowClassName(types); //[2]:查看模块
Module[] modules = assembly.GetModules();
ShowModuleName(modules); //[3]:取得类类型
Type getType = assembly.GetType("FirstReflection.ComputerFactory");
Type getTypeTwo = Type.GetType("FirstReflection.ComputerFactory");
//Console.WriteLine(getType);//输出:FirstReflection.ComputerFactory //[4]:查看该类的属性
PropertyInfo[] properties = getType.GetProperties();
ShowPropertyName(properties);//protected、private 属性未获取到 //[5]:查看实现的接口
Type[] iTypes = getType.GetInterfaces();
Console.WriteLine("查看继承的接口:");
foreach (var item in iTypes)
{
Console.WriteLine(item.Name);
} //[6]:查看类的所有方法(只获取public方法) *****
/*【get;set访问器也会被解析】*/
MethodInfo[] methods = getType.GetMethods();
Console.WriteLine("查看类的所有方法:");
foreach (var item in methods)
{
Console.WriteLine(item);
} //[6-2]:查找public方法 与getType.GetMethods()返回结果一样
MethodInfo[] publicMethods = getType.GetMethods(BindingFlags.Instance | BindingFlags.Public);
Console.WriteLine("查看类的public方法:");
foreach (var item in publicMethods)
{
Console.WriteLine(item);
} //[6-1]:查找私有方法(protected、private)
//NonPublic指反射,但找到的属性/特性是NonPublic访问的
//Instance 只是找到实例的引用对象,类似c++中的实例句柄指针(当然这个是枚举类型)
MethodInfo[] nonPublicMethods = getType.GetMethods(BindingFlags.Instance | BindingFlags.NonPublic);
Console.WriteLine("查看类的私有方法:");
foreach (var item in nonPublicMethods)
{
Console.WriteLine(item);
} //[7]:获取构造函数
ConstructorInfo[] constructors = getType.GetConstructors();
Console.WriteLine("查看类的够着方法:");
foreach (var item in constructors)
{
Console.WriteLine(item);
} //[8]:创建实例
//方法1(有问题)
object IBMcomputer = Assembly.GetAssembly(getType).CreateInstance("ComputerFactory");
//方法2
object[] constructParam = new object[] { "HP","北京" };
//带参数
object HPcomputer = Activator.CreateInstance(getType, constructParam);
//不带参数
object HPcomputer2 = Activator.CreateInstance(getType, null); //[9]:使用反射调用方法(public方法)
foreach (MethodInfo m in publicMethods)
{
if (m.Name == "displayaddress")
{//调用public void displayaddress(string address)()
object[] Param = new object[] { "北京" };
m.Invoke(HPcomputer2, Param);
}
}
//[9-1]:使用反射调用方法 (NonPublic 方法)
MethodInfo met = getType.GetMethod("privatedisplayaddress", BindingFlags.Instance | BindingFlags.NonPublic);
object[] Param2 = new object[] { "北京" };
met.Invoke(HPcomputer2, Param2); //[9-*]其他 Console.ReadLine(); }
} }
查看结果:
其中:查看类的所有方法出现 get_brand(), set_brand() 说明 get/set访问器在编译时 生成的公有方法get_brand(), set_brand()
步骤:
1.加载程序集 使用Assembly类的Load()或者LoadFrom()方法加载程序集,区别是后者需要加入程序集的后缀名,例如.DLL .exe
2.找到相应的命名空间+类 Type类型也是抽象类,只能通过GetType()或者typeof()来实现
3.获取类的方法 getMethod() 根据需要调用的方法的访问级别 public protected privat 及static传入相应的参数 方法名+ (BindingFlags.Instance | BindingFlags.Public)等
4.调用方法 MethodInfo.Invoke() 传入实例,方法的参数
基础命名空间:反射 using System.Reflection的更多相关文章
- C#反射发出System.Reflection.Emit学习
一.System.Reflection.Emit概述 Emit,可以称为发出或者产生.与Emit相关的类基本都存在于System.Reflection.Emit命名空间下.反射,我们可以取得形如程序集 ...
- 黑马程序员_Java基础:反射机制(Reflection)总结
------- android培训.java培训.期待与您交流! ---------- 反射在java中有非常重大的意义,它是一种动态的相关机制,可以于运行时加载.探知.使用编译期间完全未知的clas ...
- 基础命名空间:序列化 System.Runtime.Serialization
对象通常都有状态(state),从一个对象中抽取这种状态,不论是将它存储于某地,还是通过网络传送,这种抽取动作称为“将一个对象序列化”,而反向处理过程,从一个被序列化的状态重建一个对象即为反序列化. ...
- System.Reflection.Emit学习
C#反射发出System.Reflection.Emit学习 分享: 1 一.System.Reflection.Emit概述 Emit,可以称为发出或者产生.与Emit相关的类基本都存在于Syste ...
- 反射基础 System.Reflection
一.获取程序集Assembly 1.获取当前运行的程序集 System.Reflection.Assembly[] asm = AppDomain.CurrentDomain.GetAssemblie ...
- 【C#基础】System.Reflection (反射)
在使用.NET创建的程序或组件时,元数据(metadata)和代码(code)都存储于"自成一体"的单元中,这个单元称为装配件.我们可以在程序运行期间访问这些信息.在System. ...
- C# System.Reflection (反射)
在使用.NET创建的程序或组件时,元数据(metadata)和代码(code)都存储于“自成一体”的单元中,这个单元称为装配件.我们可以在程序运行期间访问这些信息. 在System.Reflectio ...
- C#基础|初探反射
什么是反射 我们编写的C#代码都可以编译成exe文件或dll文件.暂时先把他们叫做程序集吧,程序集中包含了很多信息.你写了一个类,类中会有字段,有属性,有方法,编译是会把这些信息保存在程序集中,暂时把 ...
- 转:.NET基础篇——反射的奥妙
反射是一个程序集发现及运行的过程,通过反射可以得到*.exe或*.dll等程序集内部的信息.使用反射可以看到一个程序集内部的接口.类.方法.字段.属性.特性等等信息.在System.Reflectio ...
随机推荐
- exc_bad_access(code=1, address=0x789870)野指针错误
原因: exc_bad_access(code=1, address=0x789870)野指针错误,主要的原因是,当某个对象被完全释放,也就是retainCount,引用计数为0后.再去通过该对象去调 ...
- CSS书写规范、顺序和命名规则
写了这么久的CSS,但大部分前端er都没有按照良好的CSS书写规范来写CSS代码,这样会影响代码的阅读体验,这里总结一个CSS书写规范.CSS书写顺序供大家参考 这些是参考了国外一些文章以及我的个 ...
- Opencv 函数
1.cvLoadImage:将图像文件加载至内存: 2.cvNamedWindow:在屏幕上创建一个窗口: 3.cvShowImage:在一个已创建好的窗口中显示图像: 4.cvWaitKey:使程序 ...
- 洛谷 P1896 互不侵犯King
P1896 [SCOI2005]互不侵犯King 题目描述 在N×N的棋盘里面放K个国王,使他们互不攻击,共有多少种摆放方案.国王能攻击到它上下左右,以及左上左下右上右下八个方向上附近的各一个格子,共 ...
- Java如何获取当前的jar包路径以及如何读取jar包中的资源
写作业的时候要输出一个record.dat文件到jar包的同级目录,但是不知道怎么定位jar包的路径.百度到的方法不很靠谱,所以在这里记录一下. 一:使用类路径 String path = this. ...
- RCP打包出来 运行 出现 JVM terminated.exit code = 13
在建立PM.product,即打包时,没有添加相应的插件,导致无法运行
- 对于JavaScript对象的prototype和__proto__的理解
一.Object和Function的关系: 刚学JavaScript的时候,看书上说JavaScript中万物皆对象,而javascript中的其他对象都是从Object继承而来,包括内置对象.瞬间觉 ...
- 函数:Python的乐高积木 - 零基础入门学习Python017
函数:Python的乐高积木 让编程改变世界 Change the world by program 相信大家小时候应该都玩过神奇的乐高积木,只要通过想象和创意,我们可以用它拼凑出很多神奇的东西. 随 ...
- ZOJ 3818 Pretty Poem
暴力模拟 细节处理很重要... #include <iostream> #include <cstring> #include <cstdio> using nam ...
- 开心菜鸟系列学习笔记------------javascript(6)
一.作用域链 1)函数的生命周期: 函数的生命周期分为创建和激活阶段(调用时),让我们详细研究它. 作用域链与一个执行上下文相关,变量 ...