zoukankan      html  css  js  c++  java
  • 反射学习:(System.Reflection)

    反射为了动态(运行时动态)
    原理:读取metadata(?)
     
    Assembly assembly = Assembly.Load("TestReflections");//反射的入口 动态加载DLL
     
    foreach (Module item in assembly.GetModules()){}//按照命名空间 决定循环次数
    //GetTypes() //类型 类的名字
     
    优点:
    依赖接口完成可配置可扩展
    可配置(不需要修改程序,只需要修改配置文件,程序执行不同的行为)
    可扩展(动态增加功能)基础可配置实现(同一个父类或者接口)
    弊端:
    方法调用时参数溢出也能通过编译
    更加耗性能
    会跳过编译器,准确性由开发者保障
     
    1.1反射的使用(依赖接口):
    Assembly assembly = Assembly.Load("TestReflections");//反射的入口 动态加载DLL
    Type type = assembly.GetType("TestReflection.DBHelper");//基础类的完整名称找出类
    object oDb = Activator.CreateInstance(type);//根据类型,创建对象
    IDBHelper IDB = (IDBHelper)oDb;//强转类型
    IDB.Query();//使用DLL的方法
     
    1.2可配置性的实现:
    1.2.1配置文件代码如:
    <appSettings>
      <add key="TestReflection" value="TestReflection,TestReflection.DBHelper"/>
    </appSettings>
    //读取配置文件信息
    string nameSpace = ConfigurationManager.AppSettings["TestReflection"];
    string[] nameSpaceArr = nameSpace.Split(',');
     
    Assembly assembly = Assembly.Load(nameSpaceArr[0]);//反射的入口 动态加载DLL
    Type type = assembly.GetType(nameSpaceArr[1]);//基础类的完整名称找出类型
    object oDb = Activator.CreateInstance(type);//根据类型,创建对象
    IDBHelper IDB = (IDBHelper)oDb;//强转类型
    IDB.Query();//使用DLL的方法
     
    1.3反射的使用(不依赖接口):
    Assembly assembly1 = Assembly.Load(nameSpaceArr[0]);//反射的入口 动态加载DLL
    Type type1 = assembly.GetType(nameSpaceArr[1]);//基础类的完整名称找出类型 即类
    object oObj = Activator.CreateInstance(type);//创建对象
    MethodInfo Query = type.GetMethod("Query");//获取指定方法(type.GetMethods();//获取类的所有方法)
    调用方法:
    1.3.1无参调用:
    Query.Invoke(oObj, null);//第一个参数:方法所在的对象 第二个参数指方法参数,详情见下
     
    1.3.2有参数的调用
    Query.Invoke(oObj, new object[] { 1, "12" });
     
    1.3.3方法重载调用方式
    MethodInfo QueryInt = type.GetMethod("Query" , new Type[] { typeof(int) });//通过第二个参数定义获取方法的参数类型,来确定获取的是哪个方法
    QueryInt.Invoke(oObj , new object[] { 1 });//调用
     
    1.3.4破坏私有函数的规则(.net访问修饰符中的private是只能本类中使用,但是通过反射可以破坏这种机制)
    MethodInfo QueryPrivate = type.GetMethod("Query", BindingFlags.Instance | BindingFlags.NonPublic);//获取私有方法
     
    1.3.5破坏单例模式(单例模式是指类只能实例化一次,即值进入一次无参构造函数。)(在上例中描述了如果破坏private的机制,在这里讲述如何破坏单例模式,即进入两次无参构造函数)
    object oObj2 = Activator.CreateInstance(type2,true);//第二个参数如果公共或非公共默认构造函数可以匹配,则为 true;如果只有公共默认构造函数可以匹配,则为 false。
     
    第三节:
    1.反射获取属性和赋值
    2.封装数据库访问层
     
    1)反射获取属性和赋值
               //反射获取属性和赋值
                //1)找到类型 (1.加载DLL 2.找到类型 3.创建对象)
                {
                    Type type3 = typeof(DBHelper);//使用typeof()找到类型 前提:知道对象类型
                    object obj3 = Activator.CreateInstance(type3);//创建对象
                    //type3.GetProperties();    //找出type3类型下的所有属性
                    foreach (var item in type3.GetProperties())
                    {
                        if (item.Name.Equals("ID"))
                            item.SetValue(obj3, 12);//给属性赋值
                        Console.WriteLine("属性名称:{0},值为{1}", item.Name, item.GetValue(obj3));//获取属性信息
                    }
                    string fileName = string.Join(",", type3.GetProperties().Select(p => string.Format("[{0}]", p.Name)));//将字符串数组按指定字符连接 返回string类型
                }
     
    2)封装数据库访问层
    实现思路:
    1.通过 类型.Name 获取到模型名称(即表名)
    2.通过 类型.GetProperties() 获取到所有的属性名称(即列名)
    3.通过 String.Join() 方法拼接列字符串
     
    注:
    实现数据查询的方法是泛型方法,因为不同的表类型不同,所以在调用的时候需要传入类型(表名)
     
    理解:
    MVC+EF 使用实体查询是 与这个类似
  • 相关阅读:
    新书推荐——《How We Test Software at Microsoft》
    [ZZ]采访与书摘:使用C#进行基于模型的软件测试和分析
    Adding Different Types of Data Sources to a Web Test
    [ZZ]为什么传统的自动化测试工具会扼杀敏捷?
    很久没有这么High了
    留图以纪念这次地震
    white——Automate windows applications
    WatiN、Wax、WatiN Test Recorder开源自动化测试框架
    绝版的T61普屏
    有感于公司搬家
  • 原文地址:https://www.cnblogs.com/xiewei123/p/10165490.html
Copyright © 2011-2022 走看看