反射基础

本文涉及的产品
云数据库 RDS SQL Server,基础系列 2核4GB
RDS SQL Server Serverless,2-4RCU 50GB 3个月
推荐场景:
简介: 反射基础

反射概念反射是由框架提供的一个类库,可以访问dll的metadata,获取的dll的信息,并使用它

反射的命名空间:using System.Reflection;

DLL里面包含了两种:

1.IL中间语言,最后变成机器码

2.metadata元数据,用于描述IL

程序结构:

Reflence引用DB.sqlserver DB.interface

DB.sqlserver引用DB.interface

cdef9656ce552e599817e52ef625f2f6_779.png

接口配置:

70d59673e657ccbeec3e7ae18428ddc5_794.png

a2ceaa5b7b658eb12ff4feaa897e15f5_800.png

普通实现方法:实例化

IDBserver dBserver =newsqlserverDBhelper();
dBserver.Query();

效果

c2313dc9fde036f927fa232824be3cd6_815.png

利用反射去实现:

//Assembly是反射的命名空间引入的类
Assembly assembly = Assembly.Load("DB.sqlserver");//动态加载dll没有后缀
Type type = assembly.GetType("DB.sqlserver.sqlserverDBhelper");//获取类型名称 传递完整类型
Object obj = Activator.CreateInstance(type);//创建对象
((sqlserverDBhelper)obj).Query();//调用方法
Console.ReadLine();


如果每增加一个类就要重新指定DLL名称,配置性不高

为了增加程序的可配置性,新建一个工厂类

如果程序需要读取配置文件就需要引入一个DLL

72a5d620641271ccfdbba467c6e7581a_893.png

引入命名空间 using System.Configuration;
配置文件添加,键值对

<appSettings>
<add key="IDBHelperType" value="DB.sqlserver,DB.sqlserver.sqlserverDBhelper"></add>
</appSettings>
publicclassObjectFactroy
{
//根据key名字获取,读取到当前项
privatestatic string IDBHelperType = ConfigurationManager.AppSettings["IDBHelperType"];
publicstatic string DLLName = IDBHelperType.Split(',')[0];//获取DLL名称
publicstatic string TypeName = IDBHelperType.Split(',')[1];//获取文件名
publicstatic IDBserver CreateInstance()
{
            Assembly assembly = Assembly.Load(DLLName);
            Type type = assembly.GetType(TypeName);//获取类型名称 传递完整类型
            Object obj = Activator.CreateInstance(type);//创建对象
return(IDBserver)obj;
}
}


有需要变更直接在配置文件更改value值

Main 函数调用

IDBserver bserver = ObjectFactroy.CreateInstance();
bserver.Query();

1.怎样让程序可配置可扩展呢,在不改变原来代码的条件下增加一个类库

7ba79e95f9156f677005049ed40d6665_911.png

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
using DB.Interface;
namespace DB.Oracle
{
publicclassOracleDBhelper: IDBserver
{
publicOracleDBhelper()
{
            Console.WriteLine("{0}被构造",this.GetType().Name);
}
publicvoidQuery()
{
            Console.WriteLine("{0},Query",this.GetType().Name);
}
}
}

修改配置文件指向新的DLL

9083a1b729ab699e0c01d22d80080432_919.png

 

上述已经配置好接口类型,第三方依赖注入的容器会帮你完成创建

实际上这就是IOC,业务逻辑层提供一个接口进行IOC的配置

b07de7c902d83da6c9305bcd9f3ec4f2_923.png

此时

1.去掉接口,反射调用方法

此时调用的是Query的无参数构造方法

Assembly assembly = Assembly.Load("DB.sqlserver");//动态加载dll没有后缀
Type type = assembly.GetType("DB.sqlserver.sqlserverDBhelper");//获取类型名称 传递完整类型
Object obj = Activator.CreateInstance(type);//创建对象
MethodInfo method = type.GetMethod("Query",newType[]{});
method.Invoke(obj,newobject[]{});

仔细观察发现这也是MVC方法响应的原理,DLL在程序最初已经编译好,MVC路径:/控制器/方法。只需要指定方法就能随意跳转页面

模仿控制器传入方法的多种参数

Assembly assembly = Assembly.Load("DB.sqlserver");//动态加载dll没有后缀
            Type type = assembly.GetType("DB.sqlserver.sqlserverDBhelper");//获取类型名称 传递完整类型
            Object obj = Activator.CreateInstance(type);//创建对象
{
{
                    MethodInfo method = type.GetMethod("Query",newType[]{});
                    method.Invoke(obj,newobject[]{});
}
{
                    MethodInfo method = type.GetMethod("Query",newType[]{typeof(string)});
                    method.Invoke(obj,newobject[]{"AB"});
}
{
                    MethodInfo method = type.GetMethod("Query",newType[]{typeof(int)});
                    method.Invoke(obj,newobject[]{20});
}
{
                    MethodInfo method = type.GetMethod("Query",newType[]{typeof(string),typeof(int)});
                    method.Invoke(obj,newobject[]{"AB",20});
}
{
                    MethodInfo method = type.GetMethod("Query",newType[]{typeof(int),typeof(string)});
                    method.Invoke(obj,newobject[]{20,"AB"});
}
}

反射最大的优点:动态加载、不需要引用命名空间

缺点:编写麻烦,避开了编译器的检查不到错误

利用反射获取属性和方法

写死的方法:


People people =newPeople()
{
     id=10,
     Name="YLC"
};
people.Property ="字段";
Console.WriteLine("id:{0}",people.id);
Console.WriteLine("Name:{0}", people.Name);
Console.WriteLine("Property:{0}", people.Property);

affaf89a7159d4999a9fd271b0786e45_1194.png


 

 

这个时候如果增加一个属性,代码又需要修改,增加100个就要打印100 Console.WriteLine();

利用反射自带的进行实现

Type type =typeof(People);
foreach(var item in type.GetProperties())//获取属性
{
                object obj = item.GetValue(people);
                Console.WriteLine("{0}:{1}",item.Name,obj);
}
foreach(var item in type.GetFields())//获取字段
{
                object obj = item.GetValue(people);
                Console.WriteLine("{0}:{1}", item.Name, obj);
}


效果相同

相关实践学习
使用SQL语句管理索引
本次实验主要介绍如何在RDS-SQLServer数据库中,使用SQL语句管理索引。
SQL Server on Linux入门教程
SQL Server数据库一直只提供Windows下的版本。2016年微软宣布推出可运行在Linux系统下的SQL Server数据库,该版本目前还是早期预览版本。本课程主要介绍SQLServer On Linux的基本知识。 相关的阿里云产品:云数据库RDS&nbsp;SQL Server版 RDS SQL Server不仅拥有高可用架构和任意时间点的数据恢复功能,强力支撑各种企业应用,同时也包含了微软的License费用,减少额外支出。 了解产品详情:&nbsp;https://www.aliyun.com/product/rds/sqlserver
相关文章
反射和反射的方法
反射和反射的方法
|
Java 数据处理 数据库
反射到底有什么作用,能帮我们干些什么呢?
反射到底有什么作用,能帮我们干些什么呢?
Java反射->什么是反射?->获取方式
Java反射->什么是反射?->获取方式
Java反射->什么是反射?->获取方式
|
JavaScript 前端开发 Java
注解与反射5.反射概述
注解与反射5.反射概述
|
算法 安全 C#
C#反射与特性(一):反射基础
C#反射与特性(一):反射基础
272 0
C#反射与特性(一):反射基础
|
Java C# C++
实战中反射的应用
实战中反射的应用
128 0
实战中反射的应用
C#反射与特性(二):探究反射
C#反射与特性(二):探究反射
212 0
|
C# 索引
C#反射与特性(八):反射操作的示例大全
C#反射与特性(八):反射操作的示例大全
353 0
下一篇
DataWorks