C#:反射以及Attribute在ORM中的应用 [转载]

C#:反射以及Attribute在ORM中的应用时间:2010-02-12 20:49:40来源:网络 作者:未知 点击:1210次

一、 反射

一、 反射

  什么是反射?

  简单点吧,反射就是在运行时动态获取对象信息的方法,比如运行时知道对象有哪些属性,方法,委托等等等等。

  反射有什么用呢?

  反射不但让你在运行是获取对象的信息,还提供运行时动态调用对象方法以及动态设置、获取属性等的能力。

  反射在ORM中有什么用呢?

  我这里所讨论的ORM实现是通过自定义Attribute的方式进行映射规则的描述的。但是我们并不知道具体哪个对象需要对应哪个表,并且这些对象是独立于我们的ORM框架的,所以我们只能通过自定义Attribute来定义映射规则,然后通过反射来动态获取这些映射规则。

  反射的实现:

  下面我们就以简单的获取对象的属性值的方式来做讨论,假设我们有类Person,其中有3个属性Name、Age,Sex。我们通过反射的方法来动态获取Person的对象的这三个属性的值。

public class Person

{

  private string _Name;

  private int _Age;

  private string _Sex;

  public string Name

  {

    get { return this._Name; }

    set { this._Name = value; }

  }

  public int Age

  {

    get { return this._Age; }

    set { this._Age = value; }

  }

  public string Sex

  {

    get { return this._Sex; }

    set { this._Sex = value; }

  }

}

测试代码如下:

static class Program

{

  [STAThread]

  static void Main()

  {

    Person person = new Person();

    person.Name = "snoopy";

    person.Age = 5;

    person.Sex = "male";

    PropertyInfo[] infos = person.GetType().GetProperties();

    Console.WriteLine("打印属性");

    foreach (PropertyInfo info in infos)

    {

      //获取属性并打印

      Console.WriteLine(info.Name + ":" + info.GetValue(person, null));

    }

    Console.WriteLine("设置Person.Name = Hellokitty");

    //设置属性,设置Name属性

    foreach (PropertyInfo info in infos)

    {

      if (info.Name == "Name")

      {

        info.SetValue(person, "Hellokitty", null);

      }

    }

    Console.WriteLine("打印属性");

    foreach (PropertyInfo info in infos)

    {

      //获取属性并打印

      Console.WriteLine(info.Name + ":" + info.GetValue(person, null));

    }

    Console.Read();

  }

}

执行结果:

上面演示了通过反射的方法来动态获取和设置对象属性的方法。但是这和ORM以及Attribute有什么关系呢?这个是我们接下来的这个部分的内容。

  上面演示了通过反射的方法来动态获取和设置对象属性的方法。但是这和ORM以及Attribute有什么关系呢?这个是我们接下来的这个部分的内容。

  二、Attribute的使用:

  Attribute中文翻译虽然也号称“属性”,但是她和对象的属性(Property)其实是完全不同的两概念。她是在运行时对对象或者对象属性、方法、委托等等进行描述的类,用于在运行时描述你的代码或者在运行时影响你的程序的行为。

  其实我们在c#的编程中经常看到Attribute,只不过我们没有注意罢了。比如Main函数前的“[STAThread]”这个其实就是一个Attribute。全程为[STAThreadAttribute]。另外指定类可序列化的[Serializable]等等。是不是都很熟悉啊?只不过平时估计没有用到,所以没有注意罢了。

  既然Attribute是类,那么她的定义方法和类就没有两样了,唯一的不同就是自定义Attribute类必须继承于System.Attribute。

  下面我们来简单定义一个描述数据库字段信息的Attribute,在此类中我们采用更省略的方式,仅仅提供“字段名”,“字段类型”:

public class DataFieldAttribute : Attribute

{

  private string _FieldName;

  private string _FieldType;

  public DataFieldAttribute(string fieldname, string fieldtype)

  {

    this._FieldName = fieldname;

    this._FieldType = fieldtype;

  }

  public string FieldName

  {

    get { return this._FieldName; }

    set { this._FieldName = value; }

  }

  public string FieldType

  {

    get { return this._FieldType; }

    set { this._FieldType = value; }

  }

}

好,我们有了自己的描述数据库字段的Attribute,那么我们现在将其应用到实际的类中。我们还是继续上面的Person类,使用方法如下:

public class Person

{

  private string _Name;

  private int _Age;

  private string _Sex;

  [DataFieldAttribute("name", "nvarchar")]

  public string Name

  {

    get { return this._Name; }

    set { this._Name = value; }

  }

  [DataFieldAttribute("age", "int")]

  public int Age

  {

    get { return this._Age; }

    set { this._Age = value; }

  }

  [DataFieldAttribute("sex", "nvarchar")]

  public string Sex

  {

    get { return this._Sex; }

    set { this._Sex = value; }

  }

}

通过自定义Attribute,我们定义了类属性和数据库字段的一一对应关系,我们对Person类的Name、Age、Sex属性都加上了Attribute的描述,指定了他们对应的字段名以及类型,其中Person.Name对应于字段name,字段类型Nvarchar...。

  三、反射和Attribute的联合使用。

  从上面的描述中,我们了解了反射,了解了Attribute,了解了ORM映射规则的定义。但是刚接触的朋友估计还是迷惑,我们怎么动态获取这些映射规则呢?

  这就需要使用反射了:

  下面的例子,我们由于对Person中的Name,Age以及SEX都增加了DataFieldAttribute的描述,这其实就是增加了O(对象)/R(关系数据库)的映射规则,下面我们就通过反射的方法来动态获取此映射规则:

static class Program

{

  [STAThread]

  static void Main()

  {

    Person person = new Person();

    person.Name = "snoopy";

    person.Age = 5;

    person.Sex = "male";

    PropertyInfo[] infos = person.GetType().GetProperties();

    object[] objDataFieldAttribute = null;

    foreach (PropertyInfo info in infos)

    {

      objDataFieldAttribute = info.GetCustomAttributes(typeof(DataFieldAttribute), false);

      if (objDataFieldAttribute != null)

      {

        Console.WriteLine(info.Name + "->数据库字段:" + ((DataFieldAttribute)objDataFieldAttribute[0]).FieldName);

      }

    }

  }

}

测试结果:

接下来的工作就是怎样根据这种方法动态地从对象中获取映射规则,动态构造Insert,Update,Delete等语句。

  四、本章总结

  本章中比较详细地介绍了反射,自定义Attribute的概念和应用,并且介绍了怎样在运行时动态获取O/R Mapping的映射规则等。当然我这里的代码仅仅是举例,而要真正实现一个ORM,我们还需要考虑的很多,比如:

  1、Person对应于哪张数据库表?

  2、Person中的PK和FK(如果有的话)怎么表示?