反射反射,程序员的快乐+反射案例:打印和Excel导出
还是那几句话:
学无止境,精益求精
十年河东,十年河西,莫欺少年穷
学历代表你的过去,能力代表你的现在,学习代表你的将来
看过设计模式的童鞋都知道:反射反射,程序员的快乐!今天我们就利用反射来制作打印和Excel报表导出,不过在进行案例之前,我们探讨下反射的基础知识:
借用别人博客的一个举例来描述反射:
B超:大家体检的时候大概都做过B超吧,B超可以透过肚皮探测到你内脏的生理情况。这是如何做到的呢?B超是B型超声波,它可以透过肚皮通过向你体内发射B型超声波,当超声波遇到内脏壁的时候就会产生一定的“回音”反射,然后把“回音”进行处理就可以显示出内脏的情况了(我不是医生也不是声学专家,不知说得是否准确^_^)。
B超利用超声波去探索人体内部的结构,反射其实和B超差不多,只不过在这里我们是利用反射去了解类的内部构造,比如类的属性,成员,方法,接口,命名空间,类的全名等
我们在使用反射前,需要引入程序集:
引入后,我们就可以进行反射的测试了,
1、首先从反射常用的 System.Type 说起,我们可以利用 System.Type 来探测类的内部接口,比如:类的属性,方法,成员,接口等
具体的用法表现为:
Type类的属性:
Name 数据类型名
FullName 数据类型的完全限定名(包括命名空间名)
Namespace 定义数据类型的命名空间名
IsAbstract 指示该类型是否是抽象类型
IsArray 指示该类型是否是数组
IsClass 指示该类型是否是类
IsEnum 指示该类型是否是枚举
IsInterface 指示该类型是否是接口
IsPublic 指示该类型是否是公有的
IsSealed 指示该类型是否是密封类
IsValueType 指示该类型是否是值类型
Type类的方法:
GetConstructor(), GetConstructors():返回ConstructorInfo类型,用于取得该类的构造函数的信息
GetEvent(), GetEvents():返回EventInfo类型,用于取得该类的事件的信息
GetField(), GetFields():返回FieldInfo类型,用于取得该类的字段(成员变量)的信息
GetInterface(), GetInterfaces():返回InterfaceInfo类型,用于取得该类实现的接口的信息
GetMember(), GetMembers():返回MemberInfo类型,用于取得该类的所有成员的信息
GetMethod(), GetMethods():返回MethodInfo类型,用于取得该类的方法的信息
GetProperty(), GetProperties():返回PropertyInfo类型,用于取得该类的属性的信息
可以调用这些成员,其方式是调用Type的InvokeMember()方法,或者调用MethodInfo, PropertyInfo和其他类的Invoke()方法。
根据上述的具体用法,我们作如下测试:
static void Main(string[] args)
{
//1、反射基本的类 获取属性及方法
Type type = typeof(Person);
Console.WriteLine("类型名:" + type.Name); Console.WriteLine("类全名:" + type.FullName); Console.WriteLine("命名空间名:" + type.Namespace); Console.WriteLine("程序集名:" + type.Assembly); Console.WriteLine("模块名:" + type.Module); Console.WriteLine("基类名:" + type.BaseType); Console.WriteLine("是否类:" + type.IsClass); Console.WriteLine("类的公共成员(Public):"); MemberInfo[] memberInfos = type.GetMembers();//得到所有公共成员
foreach (var item in memberInfos)
{
Console.WriteLine(string.Format("{0}:{1}", item.MemberType, item)); }
Console.WriteLine("类的公共属性(Public):");
PropertyInfo[] Propertys = type.GetProperties();
foreach (PropertyInfo fi in Propertys)
{
Console.WriteLine(fi.Name);
} Console.WriteLine("类的公共方法(Public):");
MethodInfo[] mis = type.GetMethods();
foreach (MethodInfo mi in mis)
{
Console.WriteLine(mi.ReturnType + " " + mi.Name);
} Console.WriteLine("类的公共字段(Public):");
FieldInfo[] fis = type.GetFields();
foreach (FieldInfo fi in fis)
{
Console.WriteLine(fi.Name);
} Console.ReadKey();
}
2、利用发射创建对象并调用相应的方法 (System.Reflection.Assembly):
namespace SJMS
{
class Program
{
static void Main(string[] args)
{
string Namespace = "SJMS.Robot";
Person report = (Person)System.Reflection.Assembly.GetExecutingAssembly().CreateInstance(Namespace + ".Robot", false, System.Reflection.BindingFlags.Default, null, null, null, null);
report.Speak();
Console.ReadKey();
}
} public abstract class Person
{
public string PersonType = "地球人";//字段
public string Name { get; set; }//属性+方法:get_Name 等
public int Asge { get; set; }//属性
public string Sex { get; set; }//属性
public abstract void Speak();//方法
} public class Student : Person
{
public string StudentNo { get; set; }
public override void Speak()
{
Console.WriteLine("我的名字是:" + Name);
} public string GetStudentNo()
{
return "";
}
}
} namespace SJMS.Robot
{
public class Robot : Person
{
public override void Speak()
{
Console.WriteLine("大家好:我是机器人。");
}
}
}
这里需要说明一下:
上述截图中的NameSpace是指命名空间,字符串 .Robot 是指类名,两者相加即构成: SJMS.Robot.Robot 这是一个类的全名,我们利用反射去查找这个类,如果找不到,则会异常并抛出未将对象引用到对象实例,因此在构造时,一定要仔细点。
呵呵
上述截图代码执行后,我们就创建了一个 SJMS.Robot.Robot 类的对象,这是一个机器人对象,如下:
创建了对象,调用说话的方法即可:
3、利用发射探测当前程序集 (System.Reflection.Assembly)
class Program
{
static void Main(string[] args)
{
//2、获取当前执行代码的程序集
Assembly assem = Assembly.GetExecutingAssembly(); Console.WriteLine("程序集全名:" + assem.FullName); Console.WriteLine("程序集的版本:" + assem.GetName().Version); Console.WriteLine("程序集初始位置:" + assem.CodeBase); Console.WriteLine("程序集位置:" + assem.Location); Console.WriteLine("程序集入口:" + assem.EntryPoint); Type[] types = assem.GetTypes();
Console.WriteLine("程序集下包含的类型:");
foreach (var item in types)
{ Console.WriteLine("类:" + item.Name);
}
Console.ReadKey();
}
}
执行如下:
OK,上述便是反射最常用的方法,在此我们进行一个反射的应用:
利用反射结合资源文件解读数据库字段
例如,我们有如下一张数据表:
Create table WeChat_User
(
Id int identity(1,1) not null primary key,
subscribe int,--用户是否订阅该公众号标识,值为0时,代表此用户没有关注该公众号,拉取不到其余信息。
openid varchar(100) not null unique,--用户的标识,对当前公众号唯一
nickname nvarchar(50),--用户的昵称
sex int default(0),--用户的性别,值为1时是男性,值为2时是女性,值为0时是未知
city nvarchar(50),--用户所在城市
province nvarchar(50),--用户所在省份
country nvarchar(50),--用户所在国家
U_language varchar(50),-- 用户的语言,简体中文为zh_CN
headimgurl varchar(300),--用户头像,最后一个数值代表正方形头像大小(有0、46、64、96、132数值可选,0代表640*640正方形头像),用户没有头像时该项为空
subscribe_time datetime,--用户关注时间
)
接触过微信公众号开发的人员都知道,上述表描述的是微信公众号粉丝信息。
根据数据表,创建对应的Model
public class SubscribeUserModel
{
public int Id { get; set; }
public int subscribe { get; set; }
public string openid { get; set; }
public string nickname { get; set; }
public int sex { get; set; }
public string language { get; set; }
public string city { get; set; }
public string province { get; set; }
public string country { get; set; }
public string headimgurl { get; set; }
public DateTime subscribe_time { get; set; }
}
根据各属性的含义。创建对应的资源文件:
创建资源文件读取类:
public class Library
{
/// <summary>
/// 获取资源文件--根据资源文件键的名字,取出对应的值
/// </summary>
/// <param name="ResourceCode">ResourceCode</param>
/// <returns></returns>
public static string GetResourceString(string ResourceCode)
{
return WeChat_User.ResourceManager.GetString(ResourceCode);
}
}
测试代码如下:
class Program
{
static void Main(string[] args)
{
//假设数据库中目前有2个粉丝
List<SubscribeUserModel> list = new List<SubscribeUserModel>();
SubscribeUserModel Model_1 = new SubscribeUserModel()
{
Id = ,
subscribe = ,
openid = "openid_1",
nickname = "隔壁老王",
sex = ,
language = "zh_CN",
city = "苏州",
province = "江苏",
country = "中国",
headimgurl = "headimgurl_1",
subscribe_time = DateTime.Now.AddDays(-)
}; SubscribeUserModel Model_2 = new SubscribeUserModel()
{
Id = ,
subscribe = ,
openid = "openid_2",
nickname = "小磨香油",
sex = ,
language = "zh_CN",
city = "郑州",
province = "河南",
country = "中国",
headimgurl = "headimgurl_2",
subscribe_time = DateTime.Now.AddDays(-)
}; list.Add(Model_1);
list.Add(Model_2);
//
Type type = typeof(SubscribeUserModel);
MemberInfo[] Properties = type.GetProperties();//得到所有公共成员
foreach (var item in list)
{
Console.WriteLine("");
Console.WriteLine("实例" + (list.IndexOf(item) + ).ToString() + "的解析如下:"); foreach (PropertyInfo Propertie in Properties)
{ string name = Propertie.Name;
object value = Propertie.GetValue(item, null);
if (value != null)
{
Console.WriteLine(Library.GetResourceString(name) + ":" + value.ToString());
}
}
} Console.ReadKey();
}
}
输出如下:
OK。上述的简单案例就这么多,下面贴出如何利用反射来进行打印和Excel的导出,由于代码比较多,因此,提供源代码下载地址:https://download.csdn.net/download/wolongbb/10273605
@陈卧龙的博客
反射反射,程序员的快乐+反射案例:打印和Excel导出的更多相关文章
- 程序员Y先生投保案例分享
大家好,我是闲鱼君.我在2018年底搞了个副业,做了保险经纪人.保险经纪人是为用户服务的第三方机构,找经纪人买保险省钱.省力.保险一次就买对,而且还能提供后续理赔服务,具体可以看我的文章<201 ...
- C# 反射Reflection——反射反射程序员的快乐
一.什么是反射 反射Reflection:System.Reflection,是.Net Framework提供的一个帮助类库,可以读取并使用metadata. 反射是无处不在的,MVC-Asp.Ne ...
- 从程序员到CTO的Java技术路线图(我爱分享)
在技术方面无论我们怎么学习,总感觉需要提升自已不知道自己处于什么水平了.但如果有清晰的指示图供参考还是非常不错的,这样我们清楚的知道我们大概处于那个阶段和水平. Java程序员 高级特性 反射.泛型. ...
- 从程序员到CTO的Java技术路线图 作者:zz563143188
在技术方面无论我们怎么学习,总感觉需要提升自已不知道自己处于什么水平了.但如果有清晰的指示图供参考还是非常不错的,这样我们清楚的知道我们大概处于那个阶段和水平. Java程序员 高级特性 反射.泛型. ...
- 从程序员到CTO的Java技术路线图 (转自安卓巴士)
在技术方面无论我们怎么学习,总感觉需要提升自已不知道自己处于什么水平了.但如果有清晰的指示图供参考还是非常不错的,这样我们清楚的知道我们大概处于那个阶段和水平. Java程序员 高级特性 反射.泛型. ...
- Java程序员职业生涯规划完整版:从程序员到CTO( 摘)
在技巧方面无论我们怎么学习,总感觉需要晋升自已不知道自己处于什么水平了.但如果有清晰的指示图供参考还是非常不错的,这样我们清楚的知道我们大概处于那个阶段和水平. Java程序员 高等特性 反射.泛型. ...
- 从程序员到CTO的Java技术路线图 JAVA职业规划 JAVA职业发展路线图 系统后台框架图、前端工程师技能图 B2C电子商务基础系统架构解析
http://zz563143188.iteye.com/blog/1877266在技术方面无论我们怎么学习,总感觉需要提升自已不知道自己处于什么水平了.但如果有清晰的指示图供参考还是非常不错的,这样 ...
- [转]从程序员到CTO的Java技术路线图
原文链接:http://zz563143188.iteye.com/blog/1877266 在技术方面无论我们怎么学习,总感觉需要提升自已不知道自己处于什么水平了.但如果有清晰的指示图供参考还是非常 ...
- 要做Java程序员 需要知道那些技术 重点有那些
一.Java程序员 高级特性 反射.泛型.注释符.自动装箱和拆箱.枚举类.可变 参数.可变返回类型.增强循环.静态导入 ...
随机推荐
- JMeter 检查点之响应断言(Response Assertion)
检查点之响应断言(Response Assertion) by:授客 QQ:1033553122 JMeter断言用于对sampler(采样器)进行额外检查,且在相同作用域中,每执行完一个samp ...
- LearnX控件漏洞挖掘与利用
前言 大学英语会用到一个 ActiveX 插件 LearnX ,最近从网上下了一个下来分析了一下,找到了一些漏洞并完成了 exploit . 虽然涉及的知识比较老旧,不过还是挺有意思的.这里分享一下整 ...
- Sqlserver精简安装选项
- Python:GUI之tkinter学习笔记3事件绑定
相关内容: command bind protocol 首发时间:2018-03-04 19:26 command: command是控件中的一个参数,如果使得command=函数,那么点击控件的时候 ...
- Linux防火墙基础与编写防火墙规则
Iptables采用了表和链的分层结构,每个规则表相当于内核空间的一个容器,根据规则集的不同用途划分为默认的四个表,raw表,mangle表,nat表,filter表,每个表容器内包括不同的规则链,根 ...
- Shell脚本应用(for、while循环语句和case分支语句)
1.for:读取不同的变量值,逐个执行同一组命令,直到取值完毕退出,变量值以空格分隔 语法: for 变量值 in 取值列表 do 命令序列 done 2.while:重复测试某个条件,成立则执 ...
- Python用户名密码登录系统(MD5加密并存入文件,三次输入错误将被锁定)及对字符串进行凯撒密码加解密操作
# -*- coding: gb2312 -*- #用户名密码登录系统(MD5加密并存入文件)及对字符串进行凯撒密码加解密操作 #作者:凯鲁嘎吉 - 博客园 http://www.cnblogs.co ...
- javascript中获取元素尺寸
Javascript获取获取屏幕.浏览器窗口 ,浏览器,网页高度.宽度的大小 屏幕可用工作区宽度:window.screen.availHeight,和浏览器无关,屏幕相关屏幕可用工作区高度:wind ...
- Dubbo+zookeeper+SpringMVC搭建最简单的分布式项目
Dubbo 是什么 一款分布式服务框架 高性能和透明化的RPC远程服务调用方案 SOA服务治理方案 Dubbo 架构流程图 Provider:服务提供方 Consumer:服务消费者 Registry ...
- [项目实践] python文件路径引用的规则,记一次使用sys.path[0]的问题,及如何区分 ../与 ./的使用场景
下面是一个获取配置的代码 def getValue(self,section,option): """ @file: string,the name of the con ...