您的位置:首页 > 编程语言 > C#

C#利用反射为属性赋值

2017-08-03 10:20 447 查看
作为一个普通程序员,相信每个人都曾经给某个model编写插入、修改方法时因为这个model的属性太多而吐槽过,当然聪明的使用例如动软代码生成器这样软件的你除外。这个时候我们一般是吐槽归吐槽,报怨归报怨,之后还不是得老老实实地一个属性一个属性的赋值。。

而一个偶然的机会看到dtcms5.0发布,无意间看到它竟然首先(当然是仅在我自己的认知中,这一发现让我对程序世界的博大精深充满了期待与敬畏)使用了一个神奇的功能,让我们程序员彻底解脱了双手,再也不用一边眼花缭乱一边生产代码了。它就是——反射原理。

什么是反射?

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

反射是.Net中获取运行时类型信息的方式,.Net的应用程序由:程序集(Assembly)、模块(Module)、类型(class)组成,而反射提供一种编程的方式,让程序员可以在程序运行期获得这几个组成部分的相关信息。

Assembly类定义和加载程序集,获得正在运行的装配件信息,也可以动态的加载装配件,以及在装配件中查找类型信息,并创建该类型的实例。

Type类可以获得对象的类型信息,此信息包含对象的所有要素:方法、构造器、属性等等,通过Type类可以得到这些要素的信息,并且调用之。

Module了解包含模块的程序集以及模块中的类等,还可以获取在模块上定义的所有全局方法或其他特定的非全局方法。

ConstructorInfo了解构造函数的名称、参数、访问修饰符(如pulic 或private)和实现详细信息(如abstract或virtual)等。使用Type的GetConstructors或 GetConstructor方法来调用特定的构造函数。

MethodInfo了解方法的名称、返回类型、参数、访问修饰符(如pulic 或private)和实现详细信息(如abstract或virtual)等。使用Type的GetMethods或GetMethod方法来调用特定的方法。

PropertyInfo了解属性的名称、数据类型、声明类型、反射类型和只读或可写状态等,获取或设置属性值。

FiedInfo了解字段的名称、访问修饰符(如public或private)和实现详细信息(如static)等,并获取或设置字段值。

EventInfo了解事件的名称、事件处理程序数据类型、自定义属性、声明类型和反射类型等,添加或移除事件处理程序。

ParameterInfo了解参数的名称、数据类型、是输入参数还是输出参数,以及参数在方法签名中的位置等。

诸如此类,还有FieldInfo、EventInfo等等,这些类都包含在System.Reflection命名空间下。

反射的作用:

1、可以使用反射动态地创建类型的实例,将类型绑定到现有对象,或从现有对象中获取类型

2、应用程序需要在运行时从某个特定的程序集中载入一个特定的类型,以便实现某个任务时可以用到反射。

3、反射主要应用与类库,这些类库需要知道一个类型的定义,以便提供更多的功能。

反射实例

那么反射如何使用呢,以下是整理的一个使用反射机制的实例可以作为参考:

1.首先要定义一个model类即如下 floor.cs,这个是不可或缺的,也是使用反射机制的要本。

using System;
namespace Model
{
/// <summary>
/// floor:实体类(属性说明自动提取数据库字段的描述信息)
/// </summary>
[Serializable]
public partial class floor
{
public floor()
{ }
#region Model
private int _id;
private string _title;
private int? _sort_id;
private int? _is_lock;
/// <summary>
/// 楼层id
/// </summary>
public int id
{
set { _id = value; }
get { return _id; }
}
/// <summary>
/// 楼层名称
/// </summary>
public string title
{
set { _title = value; }
get { return _title; }
}
/// <summary>
/// 排序
/// </summary>
public int? sort_id
{
set { _sort_id = value; }
get { return _sort_id; }
}
/// <summary>
/// 是否开启
/// </summary>
public int? is_lock
{
set { _is_lock = value; }
get { return _is_lock; }
}
#endregion Model

}
}


2.在数据访问层中即DAL中就可以使用反射来为属性赋值与取值。如下:

using System;
using System.Data;
using System.Text;
using System.Data.SqlClient;
using HWcms.DBUtility;
using System.Reflection;
using System.Collections.Generic;

namespace HWcms.DAL
{
/// <summary>
/// 数据访问类:floor
/// </summary>
public partial class floor
{
private string databaseprefix; //数据库表名前缀
public floor(string _databaseprefix)
{
databaseprefix = _databaseprefix;
}
/// <summary>
/// 增加一条数据
/// </summary>
public int Add(HWcms.Model.floor model)
{
StringBuilder strSql = new StringBuilder();
StringBuilder str1 = new StringBuilder();//数据字段
StringBuilder str2 = new StringBuilder();//数据参数
//利用反射获得属性的所有公共属性
PropertyInfo[] pros = model.GetType().GetProperties();
List<SqlParameter> paras = new List<SqlParameter>();
strSql.Append("insert into " + databaseprefix + "floor(");
foreach(PropertyInfo pi in pros)
{
//如果不是主键则追加sql字符串
if(!pi.Name.Equals("id"))
{
//判断属性值是否为空
if(pi.GetValue(model, null)!=null)
{
str1.Append(pi.Name + ",");//拼接字段
str2.Append("@" + pi.Name + ",");//声明参数
paras.Add(new SqlParameter("@" + pi.Name, pi.GetValue(model, null)));//对参数赋值
}
}
}
strSql.Append(str1.ToString().Trim(','));
strSql.Append(") values (");
strSql.Append(str2.ToString().Trim(','));
strSql.Append(") ");
strSql.Append(";select @@IDENTITY;");
object obj = DbHelperSQL.GetSingle(strSql.ToString(), paras.ToArray());
if(obj==null)
{
return 0;
}
else
{
return Convert.ToInt32(obj);
}
}
/// <summary>
/// 更新一条数据
/// </summary>
public bool Update(HWcms.Model.floor model)
{
StringBuilder strSql = new StringBuilder();
StringBuilder str1 = new StringBuilder();
//利用反射获得属性的所有公共属性
PropertyInfo[] pros = model.GetType().GetProperties();
List<SqlParameter> paras = new List<SqlParameter>();
strSql.Append("update " + databaseprefix + "floor set ");
foreach(PropertyInfo pi in pros)
{
//如果不是主键则追加sql字符串
if(!pi.Name.Equals("id"))
{
//判断属性值是否为空
if(pi.GetValue(model,null)!=null)
{
str1.Append(pi.Name + "=@" + pi.Name + ",");
paras.Add(new SqlParameter("@" + pi.Name, pi.GetValue(model, null)));
}
}
}
strSql.Append(str1.ToString().Trim(','));
strSql.Append(" where id=@id");
paras.Add(new SqlParameter("@id", model.id));
return DbHelperSQL.ExecuteSql(strSql.ToString(), paras.ToArray()) > 0;
}
/// <summary>
/// 得到一个对象实体
/// </summary>
public HWcms.Model.floor GetModel(int id)
{
StringBuilder strSql = new StringBuilder();
StringBuilder str1 = new StringBuilder();
Model.floor model = new Model.floor();
PropertyInfo[] pros = model.GetType().GetProperties();
foreach(PropertyInfo p in pros)
{
str1.Append(p.Name + ",");
}
strSql.Append("select top 1 " + str1.ToString().Trim(','));
strSql.Append(" from " + databaseprefix + "floor");
strSql.Append(" where id=@id");
SqlParameter[] param = {
new SqlParameter("@id", SqlDbType.Int, 4)
};
param[0].Value = id;
DataTable dt = DbHelperSQL.Query(strSql.ToString(), param).Tables[0];
if(dt.Rows.Count>0)
{
return DataRowToModel(dt.Rows[0]);
}
else
{
return null;
}
}
/// <summary>
/// 将对象转换为实体
/// </summary>
public HWcms.Model.floor DataRowToModel(DataRow row)
{
HWcms.Model.floor model = new HWcms.Model.floor();
if (row != null)
{
//利用反射获取属性的所有公共属性
Type modelType = model.GetType();
for(int i=0;i<row.Table.Columns.Count;i++)
{
//查找实体是否存在列表相同的公共属性
PropertyInfo proInfo = modelType.GetProperty(row.Table.Columns[i].ColumnName);
if(proInfo!=null&& row[i]!=DBNull.Value)
{
//用索引值设置属性值
proInfo.SetValue(model, row[i], null);
}
}
}
return model;
}
}
}


刚刚使用了反射,感觉这个功能真的很走心,为了更多的了解反射,也去搜索了很多反射很多相关的知识,在一篇文章中看到反射并不是万能的,使用反射反而会降低程序的性能,导航速度非常慢,这一下了解让我意识到任何方法的优劣并不能以偏概全,可以适当的使用了解,但不能太过于依赖。以下是看来的关于反射的性能方面的知识以助了解:

反射的性能:

1、现实应用程序中很少有应用程序需要使用反射类型

2、使用反射动态绑定需要牺牲性能

3、有些元数据信息是不能通过反射获取的

使用反射来调用类型或者触发方法,或者访问一个字段或者属性时clr 需要做更多的工作:校验参数,检查权限等等,所以速度是非常慢的。所以尽量不要使用反射进行编程,对于打算编写一个动态构造类型(晚绑定)的应用程序,可以采取以下的几种方式进行代替:

1、通过类的继承关系。让该类型从一个编译时可知的基础类型派生出来,在运行时生成该类型的一个实例,将对其的引用放到其基础类型的一个变量中,然后调用该基础类型的虚方法。

2、通过接口实现。在运行时,构建该类型的一个实例,将对其的引用放到其接口类型的一个变量中,然后调用该接口定义的虚方法。

3、通过委托实现。让该类型实现一个方法,其名称和原型都与一个在编译时就已知的委托相符。在运行时先构造该类型的实例,然后在用该方法的对象及名称构造出该委托的实例,接着通过委托调用你想要的方法。这个方法相对与前面两个方法所作的工作要多一些,效率更低一些。

这是一篇如何利用委托优化反射的文章:

http://www.cnblogs.com/xinaixia/p/5777886.html

PS:利用反射机制实现添加功能时,一定要记得对那些DateTime类型的字段进行初始化,否则会报错。
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签:  反射机制 反射