反射反射,程序员的快乐+反射案例:打印和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程序员 高级特性 反射.泛型.注释符.自动装箱和拆箱.枚举类.可变 参数.可变返回类型.增强循环.静态导入 ...
随机推荐
- 安卓开发_浅谈DatePicker(日期选择器)
DatePicker继承自FrameLayout类,日期选择控件的主要功能是向用户提供包含年.月.日的日期数据并允许用户对其修改.如果要捕获用户修改日期选择控件中的数据事件,需要为DatePicker ...
- Android Studio:Support Library依赖包的版本号
当我们用RecyclerView时,如果想用某一个特定的版本,怎样才能知道版本号呢?如果自己的笔记本中用过这个库,那么会保存在本地硬盘中. Android自身依赖包的版本号本地存放路径: 没有用过该 ...
- Android Java语法学习
Activity中有一个名称叫onCreate的方法.该方法是在Activity创建时被系统调用,是一个Activity生命周期的开始. onCreate方法的参数savedInstanceState ...
- 【Java入门提高篇】Day24 Java容器类详解(七)HashMap源码分析(下)
前两篇对HashMap这家伙的主要方法,主要算法做了一个详细的介绍,本篇主要介绍HashMap中默默无闻地工作着的集合们,包括KeySet,values,EntrySet,以及对应的迭代器:HashI ...
- jpa 联表查询 返回自定义对象 hql语法 原生sql 语法 1.11.9版本
-----业务场景中经常涉及到联查,jpa的hql语法提供了内连接的查询方式(不支持复杂hql,比如left join ,right join). 上代码了 1.我们要联查房屋和房屋用户中间表,通过 ...
- Python:BeautifulSoup移除某些不需要的属性
很久之前,我看到一个问题,大概是: 他爬了一段html,他获取下了所需的部分(img标签部分),但是不想保留img标签的某些属性, 比如 <img width="147" h ...
- matlab练习程序(FAST特征点检测)
算法思想:如果一个像素与它邻域的像素差别较大(过亮或过暗) , 那它更可能是角点. 算法步骤: 1.上图所示,一个以像素p为中心,半径为3的圆上,有16个像素点(p1.p2.....p16). 2.定 ...
- python第五十七天------补上笔记
direct_client:广播接收 #!/usr/bin/env python #_*_coding:utf-8_*_ import pika,time,sys connection = pika. ...
- MD5生成
/// <summary> /// 32位MD5加密 /// </summary> /// <param name="input"></p ...
- Appium1.9.1 部署及结果检验
1.官网下载最新的 appium 2.点击 Download Appium 3.选择适用于自己操作系统的版本,我的是 windows版本,就选择如下红圈起的 4.点击安装,一直点 下一步 直到提示安装 ...