小白详细解析C#反射特性实例

时间:2022-05-02 18:58:39

  套用MSDN上对于反射的定义:反射提供了封装程序集、模块和类型的对象(Type 类型)。可以使用反射动态创建类型的实例,将类型绑定到现有对象,或从现有对象获取类型并调用其方法或访问其字段和属性。如果代码中使用了属性,可以利用反射对它们进行访问。

地址:https://msdn.microsoft.com/zh-cn/library/ms173183(VS.80).aspx

  贴上示例代码:

首先程序入口代码Program

using System;
using System.Collections.Generic;
using System.Linq;
using System.Threading.Tasks;
using System.Windows.Forms; namespace WindowsFormsApp1
{
static class Program
{
/// <summary>
/// 应用程序的主入口点。
/// </summary>
[STAThread]
static void Main()
{
UserInf userss = new UserInf();
userss.U_UserID = "aw12311";
userss.U_Psw = "";
userss.U_UserName = "aw";
userss.U_City = "武汉";
userss.U_Popedom = ;
userss.U_Sex = ;
userss.U_BirthTime = ;
userss.U_AddDataTime = DateTime.Now; DateIsTableAttribute<UserInf> t = new DateIsTableAttribute<UserInf>(); t.insertDate(userss); Application.EnableVisualStyles();
Application.SetCompatibleTextRenderingDefault(false);
Application.Run(new Form1());
}
}
}

将要被反射的程序类UserInf

using System;
using System.Collections.Generic;
using System.Data;
using System.Linq;
using System.Text;
using System.Threading.Tasks; namespace WindowsFormsApp1
{
[Table("Consumers")]
public class UserInf
{
private string _UserID;
/// <summary>
/// 登陆ID
/// </summary>
[Field("ConsumerID", DbType.String, )]
public string U_UserID
{
get { return _UserID; }
set { _UserID = value; }
} private string _Psw;
/// <summary>
/// 登陆密码
/// </summary>
[Field("ConsumerPwd", DbType.String, )]
public string U_Psw
{
get { return _Psw; }
set { _Psw = value; }
} private string _UserName;
/// <summary>
/// 用户别称
/// </summary>
[Field("ConsumerName", DbType.String, )]
public string U_UserName
{
get { return _UserName; }
set { _UserName = value; }
} private string _City;
/// <summary>
/// 所住城市
/// </summary>
[Field("UserCity", DbType.String, )]
public string U_City
{
get { return _City; }
set { _City = value; }
} private int _Popedom;
/// <summary>
/// 权限
/// </summary>
[Field("popedom", DbType.Int32, )]
public int U_Popedom
{
get { return _Popedom; }
set { _Popedom = value; }
} private DateTime _AddDataTime;
/// <summary>
/// 注册时间
/// </summary>
[Field("addDataTime", DbType.Date, )]
public DateTime U_AddDataTime
{
get { return _AddDataTime; }
set { _AddDataTime = value; }
} private int _Sex;
/// <summary>
/// 性别
/// </summary>
[Field("Sex", DbType.Int32, )]
public int U_Sex
{
get { return _Sex; }
set { _Sex = value; }
} private int _BirthTime;
/// <summary>
/// 出身日期;
/// </summary>
[Field("BirthTime", DbType.String, )]
public int U_BirthTime
{
get { return _BirthTime; }
set { _BirthTime = value; }
}
}
}

将要被反射的程序类中引伸的自定义特性类:包括预定义特性[AttributeUsage(...)]和自定义特性类具体内容

  1、自定义特性类TableAttribute

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text; namespace WindowsFormsApp1
{
[AttributeUsage(AttributeTargets.Class, AllowMultiple = false, Inherited = false)] public class TableAttribute : Attribute
{
private string _TableName; /// <summary>
/// 映射的表名
/// </summary>
public string TableName
{
get { return _TableName; }
} /// <summary>
/// 定位函数映射表名;
/// </summary>
/// <param name="table"></param>
public TableAttribute(string table)
{
_TableName = table;
}
}
}

  2、自定义特性类FieldAttribute

using System;
using System.Collections.Generic;
using System.Data;
using System.Linq;
using System.Text;
using System.Threading.Tasks; namespace WindowsFormsApp1
{
[AttributeUsage(AttributeTargets.Property, AllowMultiple = false, Inherited = false)]
public class FieldAttribute : Attribute
{
private string _Fields;
/// <summary>
/// 字段名称 keleyi.com
/// </summary>
public string Fields
{
get { return _Fields; }
} private DbType _Dbtype;
/// <summary>
/// 字段类型
/// </summary>
public DbType Dbtype
{
get { return _Dbtype; } } private int _ValueLength;
/// <summary>
/// 字段值长度
/// </summary>
public int ValueLength
{
get { return _ValueLength; }
}
/// <summary>
/// 构造函数
/// </summary>
/// <param name="fields"> 字段名</param>
/// <param name="types"> 字段类型</param>
/// <param name="i"> 字段值长度</param>
public FieldAttribute(string fields, DbType types, int i)
{
_Fields = fields;
_Dbtype = types;
_ValueLength = i;
}
}
}

最后使用Type类型将封装的程序集的类型绑定到现有对象,并利用当前对象访问、检测和修改封装的程序集中特性

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Data;
using System.Reflection; namespace WindowsFormsApp1
{
public class DateIsTableAttribute<T>
{
public string insertDate(T types)
{
string cmdtxt = "insert into ";
string cmdparVar = null;
Type userAttu = types.GetType();
TableAttribute tables = (TableAttribute)userAttu.GetCustomAttributes(false)[];//UserInf中的Table结点
cmdtxt += tables.TableName + "(";
PropertyInfo[] info = userAttu.GetProperties();
foreach (PropertyInfo prs in info)
{
object[] attu = prs.GetCustomAttributes(false);
foreach (Attribute abute in attu)
{
if (abute is FieldAttribute)
{
FieldAttribute midle = abute as FieldAttribute;
cmdtxt += midle.Fields + ",";
object obj = prs.GetGetMethod().Invoke(types, null);
if (midle.Dbtype == DbType.Int32)
cmdparVar += obj + ",";
else
cmdparVar += "'" + obj + "',";
}
}
}
cmdparVar = cmdparVar.Substring(, cmdparVar.Length - );
cmdtxt = cmdtxt.Substring(, cmdtxt.Length - ) + ")";return cmdtxt;
}
}
}

由于我使用窗口程序测试,没有设置测试结果的显示。可以将程序使用控制台实现,将cmdparVar和cmdtxt显示出来。

断点运行可以看到

cmdparVar值:'aw12311','123','aw','武汉',1,'2018/8/18 16:43:17',1,'19900114'
cmdtxt值:insert into Consumers(ConsumerID,ConsumerPwd,ConsumerName,UserCity,popedom,addDataTime,Sex,BirthTime)
利用Type类型读取到了传入程序集中的变量值。 程序结构分析:
Program为程序入口
UserInf为将要被反射的类:自定义了一个类属性“Table”,Table类属性附加了一个参数。自定义了变量属性“Field”,Field变量属性附加了3个参数。
自定义特性类TableAttribute:对UserInf的类属性“Table”重新构建并获取自定义附加参数。
自定义特性类FieldAttribute:对UserInf的变量属性“Field”重新构建并获取自定义附加参数。
程序过程分析:
一、入口程序往UserInf的各个变量进行赋值。
二、将UserInf传入模板类DateIsTableAttribute,模板类DateIsTableAttribute实现用Type反射UserInf属性。 以下进行DateIsTableAttribute详细解析:
    public class DateIsTableAttribute<T>  //实例化模板类DateIsTableAttribute时,传入T类型
{
public string insertDate(T types) //用传入T类型新建types变量
{
string cmdtxt = "insert into ";
string cmdparVar = null;
Type userAttu = types.GetType(); //获取types的类型(即传入的T类型)的当前实例传给Type类型userAttu,即反射的定义
TableAttribute tables = (TableAttribute)userAttu.GetCustomAttributes(false)[0];//获取已实例化Type类型userAttu中的直接结点并调用派生自attribute的类TableName(传入为自定义的特性)
        //并强制转换为TableAttribute类型
        //转换成TableAttribute类型是为了更好调用自定义的附加参数,比如下面tables的TableName值。
cmdtxt += tables.TableName + "(";
PropertyInfo[] info = userAttu.GetProperties();//获取传入的当前实例中的所有公共属性
foreach (PropertyInfo prs in info)//遍历所有带有attribute公共属性的元数据变量
{
object[] attu = prs.GetCustomAttributes(false);//获取此元数据中派生自Attribute类的结点
foreach (Attribute abute in attu)//遍历元数据中的自定义结点
{
if (abute is FieldAttribute)//若为Field结点
{
FieldAttribute midle = abute as FieldAttribute;//强制转换当前attibute属性
cmdtxt += midle.Fields + ",";//读取属性值
object obj = prs.GetGetMethod().Invoke(types, null);
if (midle.Dbtype == DbType.Int32)
cmdparVar += obj + ",";
else
cmdparVar += "'" + obj + "',";
}
}
}
cmdparVar = cmdparVar.Substring(0, cmdparVar.Length - 1);
cmdtxt = cmdtxt.Substring(0, cmdtxt.Length - 1) + ")";return cmdtxt;
}
}
}

用到的几个函数接口:
types.GetType():获取types当前实例的Type
Type.GetCustomAttributes(false)[0] :获取Type类型参数第一个自定义特性结点,返回为自定义特性结点的特征值,比如名称、类型、长度等
Type.GetCustomAttributes(false) 获取Type类型参数所有自定义特性结点,返回为所有自定义特性结点的特征值(比如名称、类型、长度等)组,示例中将返回的自定义特征值强制转换为自定义特性类
链接:https://msdn.microsoft.com/zh-cn/library/system.type.getcustomattributes.aspx
TableAttribute或FieldAttribute,可以更方便读取其中的自定义附加参数
Type.GetProperties() 返回当前Type的所有公共属性
链接:https://technet.microsoft.com/zh-cn/windowsserver/1zkfcy34
PropertyInfo.GetGetMethod() 返回此属性公共get访问器
链接:https://technet.microsoft.com/zh-cn/windowsserver/1zkfcy34
Method().Invoke(obj, null);反射执行该类型示例方法,obj为方法所属类型实例
http://blog.sina.com.cn/s/blog_976ba8a501010y5k.html 延伸:
[AttributeUsage(AttributeTargets.Property, AllowMultiple = false, Inherited = false)]
https://blog.csdn.net/honey199396/article/details/51316754

typeof(Animal).IsAssignFrom(typeof(Dog)) 他返回true的条件是 Dog类直接或间接的实现了Animal类;继承也可以

typeof(Dog).IsSubClassOf(typeof(Animal)) 他返回true的条件是Dog类是Animal的子类