反射ORM
七章 反射
1节
获取dll文件中的type-----------------------------------------------------------------------------------------------------------------------(*)
//程序集
程序集是.net中的概念,.dll和.exe都属于程序集
AssemblyInfo.cs是当前程序集的一些信息(版本号)
程序集包含:类型元数据(描述在代码中定义的每一类型和成员,二进制形式)、程序集元数据(程序集清单、版本号、名称等)、IL代码(这些都被装在exe或dll中)、资源文件。
每个程序集都有自己的名称版本等信息。这些信息通过AssemblyInfo.cs文件来自己定义。
//程序集得好处:
程序中只引用必须的程序集,减小程序的尺寸。
程序集可以封装一些代码,只提供必要的访问接口。
//反射
反编译工具和智能提示都是通过反射来实现的。
反射:动态获取程序集中的元数据来操作类型的
Type类是实现反射的一个重要的类,通过它我们可以获取类中的所有的信息,包括方法、属性等。可以动态调用属性和方法。Type是对类的描述
//如何获取一个类中的所有属性
//在本类中
//1
//Type tp = typeof(Person); //获得当前程序集指定公共类----------------------------(*)
//2
Person per = new Person();
Type tp = per.GetType();//也可以获得当前实例的公共类---------------------------(*)
MethodInfo[] methes = tp.GetMethods(); //获得当前公共类的所有公共方法
for (int i = 0; i < methes.Length;i++ )
{
Console.WriteLine(methes[i].Name);
}
PropertyInfo[] ps = tp.GetProperties();//获得当前公共类的所有公共属性
foreach(PropertyInfo p in ps)
{
Console.WriteLine(p.Name);
}
MyClass类库 (不需要引用,直接获得程序集)
//在外部类中
string path = @"G:\RuPeng_YZK_150107\Console_NETStronger_Chapter_7\MyClasses\bin\Debug\MyClasses.dll";
Assembly ass = Assembly.LoadFile(path);//加载指定路径上的程序集信息文件------------(*)
Type[] tps = ass.GetExportedTypes();//获得此程序集信息中定义的类型--------------(*)
foreach(Type tp in tps)
{
//Console.WriteLine(tp.Name);
MethodInfo[] methes = tp.GetMethods(); //获得该公共类中的所有方法
foreach (MethodInfo meth in methes)
{
Console.WriteLine(meth.Name);
}
}
2节
三个比较重要的方法------------------------------------------------------------------------------------------------------(*)
//第二中获得程序集的所有公共类型,指定公共类型
string path = @"G:\RuPeng_YZK_150107\Console_NETStronger_Chapter_7\MyThreeClasses\bin\Debug\MyThreeClasses.dll";
Assembly ass = Assembly.LoadFile(path);//从指定程序集路径获得程序集
Type tpPerson = ass.GetType("MyThreeClasses.Person");//获得指定的程序集类型名称的类型实例
Type tpStudent = ass.GetType("MyThreeClasses.Student");
//Console.WriteLine(tp.Name);
bool flag = tpPerson.IsAssignableFrom(tpStudent);//当前类型的实例是否可以用指定类型的实例进行赋值--true
bool flag1 = tpStudent.IsAssignableFrom(tpPerson);//false--因为Student是Person的子类
bool flag2 = tpPerson.IsSubclassOf(tpStudent);//当前类是否是指定类派生的,即是否是指定类的子类
bool flag3 = tpStudent.IsSubclassOf(tpPerson);
object obj = Activator.CreateInstance(tpStudent);//创建指定类型的一个实例
bool flag4 = tpPerson.IsInstanceOfType(obj);//指定对象是否是当前类的实例
Type tpAnimal = ass.GetType("MyThreeClasses.Animal");
bool flag5 = tpAnimal.IsAbstract;//属性,指定类是否是抽象类
Console.WriteLine(flag5);
3节
动态调用程序集中的方法------------------------------------------------------------------------------------------------(*)
//动态调用(dynamic call)
//需要先获得类型的方法,然后再调用这个方法
//Person per = new Person();
//Type tp = per.GetType();
Type tp = typeof(Person);
MethodInfo meth = tp.GetMethod("SayHello",new Type[]{typeof(string)});//获得这个类型中指定名称的公共方法
object obj = Activator.CreateInstance(tp); //创建一个指定类型的实例
meth.Invoke(obj,new object[]{"....很想"}); //使用指定参数调用当前实例的方法 //没有参数就是null
Console.ReadKey();
class Person
{
public void SayHello(string str)
{
Console.WriteLine("hello,想我了吗?"+str);
}
}
4节
反射插件(Reflecting the plug-in)----------------------------------------------------------------------------------------(*)
//做一个小写转大写的插件
主程序--->搜索dll文件
获得所有程序集
获取Type
是否实现了插件的规范(用接口方式命名这个接口)
菜单(文件 编辑 视图 帮助)
TextBox(Dock)
//寻找当前程序集路径,在其所在目录的文件下,搜索指定程序集dll
//主程序
//窗口加载时,添加插件Editplus
private void Form1_Load(object sender, EventArgs e)
{
//获得插件的路劲
string path = Path.Combine(Path.GetDirectoryName(Assembly.GetExecutingAssembly().Location), "Lib");
//搜索目录中与指定模式匹配的文件集dll文件
string[] files = Directory.GetFiles(path, "*.dll");
foreach(string file in files)
{
//加载指定路径中程序集的内容
Assembly ass = Assembly.LoadFile(file);
//获得该程序集中所有的类型
Type[] tps = ass.GetTypes();
//获得接口的类型
Type ieditplus = typeof(IEditplus);
//插件程序集的所有类型中实现接口的类型
foreach(Type tp in tps)
{
//如果接口类型的实例可以由某一类型的实例赋予,且这一类型又不是抽象类,就可以创建这一类型的实例,去实例化获得这个接口
if(ieditplus.IsAssignableFrom(tp) && !tp.IsAbstract )
{
IEditplus iedit = (IEditplus)Activator.CreateInstance(tp);
//这个接口的插件名称就可以附加到菜单的下拉项中
ToolStripItem tsi = tsmView.DropDownItems.Add(iedit.Name);
//菜单下拉项tsi的点击事件,执行接口中的"小写转大写方法",所以需要传接口,可以用控件的tag存接口
tsi.Tag = iedit;//可以用当前点击控件的tag存接口
tsi.Click += new EventHandler(tsi_Click);
}
}
}
}
private void tsi_Click(object sender, EventArgs e)
{
//需要获得当前控件
ToolStripItem tsi = sender as ToolStripItem;
//需要获得这个接口
IEditplus iedit = tsi.Tag as IEditplus;
//需要执行接口中的方法
txt.Text = iedit.LittleToLarge(txt);
}
//插件的接口类---(主程序和子程序都需要添加对接口的引用)
namespace ClassLibary_Notice
{
public interface IEditplus
{
string Name { get; } //这个插件 接口的名称,是只读、不能修改的
string LittleToLarge(TextBox tb); //插件 接口中的小写转大写方法---文本框需要添加System.Windows.Forms的引用
}
}
//插件(接口类的子类,子程序)---(需要把插件放入主程序的Lib中,使主程序可以找到实现了接口的类型)
namespace ClassLibrary_Notice_Substring.cs
{
public class Editplus:IEditplus
{
public string Name
{
get { return "小写转大写"; }
}
public string LittleToLarge(System.Windows.Forms.TextBox tb)
{
return tb.Text.ToUpper();
}
}
}
反射ORM的更多相关文章
- 反射ORM 三层(for oracle)
sql server and oracle 不同之处只在于: 1·参数@和: 2·自增和序列 3·oracle使用了存储过程以获得当前插入数据的ID DAL层的ORM using Oracle.Dat ...
- 反射ORM 三层(for sql server/mysql)
sql server and oracle 不同之处只在于: 1·参数@和: 2·自增和序列 RPROM.cs //(写错愕,应该是RPORM) RPROM.Insert(p1); //需求:DBS中 ...
- orm 小结
1. 销售注册,登录系统 - 用户表 2.销售添加客户信息,成为销售的私户 - 客户表 3. 销售固定时间跟进客户 - 跟进记录表 4. 客户报名 - 报名记录表 - 班级表(必须 ...
- 巨蟒django之CRM3 添加和编辑客户&&公户和私户的展示和转换
昨日内容回顾: day66 1. 内容回顾 1. 数据的展示 数据通过ORM查询出来 对象列表 QuerySet 1. 普通的字段 对象.字段名 ——> 数据库中的值 2. choices (( ...
- 一头扎进 JAVA
硅不可 吉米 JAVA 基础 -- 基础不牢,地动山摇 子类应该比 父类更为 开放 (public protected default private) 子类方法不能比父类抛出更高异常( 可以为父类方 ...
- ORM框架通过映射(反射)获取数据库的数据
ORM(Object Relational Mapping)框架采用元数据来描述对象一关系映射细节,元数据一般采用XML格式,并且存放在专门的对象一映射文件中.只要提供了持久化类与表的映射关系,ORM ...
- C#基础系列:实现自己的ORM(反射以及Attribute在ORM中的应用)
反射以及Attribute在ORM中的应用 一. 反射什么是反射?简单点吧,反射就是在运行时动态获取对象信息的方法,比如运行时知道对象有哪些属性,方法,委托等等等等.反射有什么用呢?反射不但让你在运行 ...
- ORM中去除反射,添加Expression
之前接触了别人的ORM框架,感觉牛掰到不行,然后试着自己来写自己的ORM. 最初从园子里找到其他人写的反射的例子: List<PropertyInfo> pis = typeof(T).G ...
- 利用抽象、多态实现无反射的绿色环保ORM框架
最近一直在忙新公司的基础库建设,对系统架构.开发框架及快速开发平台的设计实施都积累了一定的实践经验. 一般的中小型的软件开发公司,如果按照技术储备来衡量软件项目的技术含量的评定依据是可行的.但如果光是 ...
随机推荐
- 容器rootfs
挂载在容器根目录上.用来为容器进程提供隔离后执行环境的文件系统,就是所谓的“容器镜像”: 它还有一个更为专业的名称:rootfs (根文件系统). 所以,一个最常见的rootfs,或者说容器镜像,会包 ...
- codeforces Codeforces Round #318 div2 A. Bear and Elections 【优先队列】
A. Bear and Elections time limit per test 1 second memory limit per test 256 megabytes input standar ...
- hive学习1(hive基本概念)
hive基本概念 hive简介 hive是什么 Hive是基于Hadoop的一个数据仓库工具,可以将结构化的数据文件映射为一张数据库表,并提供类SQL查询功能. 为什么使用hive 1)简单易上手.提 ...
- lua 写逻辑打印log时,打印到一半后停止不再打印,程序停止
问题描述:ubuntu下用lua开发游戏电子邮件模块,自己测试时向用户推送100封,而用户最多只能有50封.这是调用sysdelete删除一些邮件.当打印log时,打印到一半后程序中途停止.将打印lo ...
- Codeforces Round #374 (Div. 2) A , B , C 水,水,拓扑dp
A. One-dimensional Japanese Crossword time limit per test 1 second memory limit per test 256 megabyt ...
- numpy array或matrix的交换两行
A[j,:] = A[maxindex,:] # 注意这样是一个很低级的错误!这样只是赋值 我们很容易想起python中的两个值交换一句搞定不用引入中间变量 a, b = b, a 但在numpy的a ...
- zeptojs的一些别人的博客
http://www.css88.com/doc/zeptojs/ http://blog.163.com/litianyichuanqi@126/blog/static/11597944120142 ...
- 《Advanced Bash-scripting Guide》学习(八):从一个目录移动整个目录树到另一个目录
本文所选的例子来自于<Advanced Bash-scripting Gudie>一书,译者 杨春敏 黄毅 ABS书上的例子: 从一个目录移动整个目录树到另一个目录 #!/bin/bash ...
- hdu2665 主席树(可持久化线段树)
题意:给定一个数组,每次查询第l到r区间的第k大值 解法嘛,当然是主席树,主席树即可持久化线段树,什么叫可持久化呢,就是指能够访问历史版本的数据结构,那么对于某些只能离线处理的题目强制在线之后 ,可以 ...
- android:descendantFocusability用法简析(转)
开发中很常见的一个问题,项目中的listview不仅仅是简单的文字,常常需要自己定义listview,自己的Adapter去继承BaseAdapter,在adapter中按照需求进行编写,问题 ...