还是那几句话:
学无止境,精益求精
十年河东,十年河西,莫欺少年穷
学历代表你的过去,能力代表你的现在,学习代表你的将来
看过设计模式的童鞋都知道:反射反射,程序员的快乐!今天我们就利用反射来制作打印和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.****.net/download/wolongbb/10273605
@陈卧龙的博客