魔兽世界1.12德鲁伊宏:.Net 中的反射(序章)

来源:百度文库 编辑:中财网 时间:2024/04/29 12:42:08

.Net 中的反射(序章) - Part.1
http://www.cnblogs.com/JimmyZhang/archive/2008/01/27/Reflection-Part1.html

引言

反射是.Net提供给我们的一件强力武器,尽管大多数情况下我们不常用到反射,尽管我们可能也不需要精通它,但对反射的使用作以初步了解在日后的开发中或许会有所帮助。

反射是一个庞大的话题,牵扯到的知识点也很多,包括程序集、自定义特性、泛型等,想要完全掌握它非常不易。本文仅仅对反射做一个概要介绍,关于它更精深的内容,需要在实践中逐渐掌握。本文将分为下面几个部分介绍.Net中的反射:

  1. 序章,我将通过一个例子来引出反射,获得对反射的第一印象。
  2. 反射初步、Type类、反射普通类型。(修改中,近期发布...)
  3. 反射特性(Attribute)。
  4. xxxx (待定)
  5. ...

序章

如果你还没有接触过反射,而我现在就下一堆定义告诉你什么是反射,相信你一定会有当头一棒的感觉。我一直认为那些公理式的定义和概念只有在你充分懂得的时候才能较好的发挥作用。所以,我们先来看一个开发中常遇到的问题,再看看如何利用反射来解决:

在进行数据库设计的过程中,常常会建立一些基础信息表,比如说:全国的城市,又或者订单的状态。假设我们将城市的表,起名为City,它通常包含类似这样的字段:

Id Int Identity(1,1) 城市Id
Name Varchar(50) 城市名称
ZIP Varchar(10) 城市邮编
... // 略

这个表将供许多其他表引用。假如我们在建立一个酒店预订系统,那么酒店信息表(Hotel)就会引用此表,用CityId字段来引用酒店所在城市。对于城市(City)表这种情况,表里存放的记录(城市信息)是不定的,意思就是说:我们可能随时会向这张表里添加新的城市(当某个城市的第一家酒店想要加入预订系统时,就需要在City表里新添这家酒店所在的城市)。此时,这样的设计是合理的。

1.建表及其问题

我们再看看另外一种情况,我们需要标识酒店预订的状态:未提交、已提交、已取消、受理中、已退回、已订妥、已过期。此时,很多开发人员会在数据库中建立一张小表,叫做BookingStatus(预订状态),然后将如上状态加入进去,就好像这样:

如同城市(City)表一样,在系统的其他表,比如说酒店订单表(HotelOrder)中,通过字段StatusId引用这个表来获取酒店预订状态。然而,几个月以后,虽然看上去和城市表的用法一样,结果却发现这个表只在数据库做联合查询或者 只在程序中调用,却从来不做修改,因为预订流程确定下来后一般是不会变更的。在应用程序中,也不会给用户提供对这个表记录的增删改操作界面。

而在程序中调用这个表时,经常是这种情况:我们需要根据预订状态对订单列表进行筛选。此时通常的做法是使用一个下拉菜单(DropDownList),菜单的数据源(DataSource),我们可以很轻易地通过一个SqlDataReader获得,我们将DropDownList的文本Text设为Status字段,将值Value设为Id字段。

此时,我们应该已经发现问题:

  1. 如果我们还有航班预订、游船预订,或者其他一些状态,我们需要在数据库中创建很多类似的小表,造成数据库表的数目过多。
  2. 我们使用DropDownList等控件获取表内容时,需要连接到数据库进行查询,潜在地影响性能。

同时,我们也注意到三点:

  1. 此表一般会在数据库联合查询中使用到。假设我们有代表酒店订单的HotelOrder表,它包含代表状态的StatusId字段,我们的查询可能会像这样:Select *, (Select Status From BookingStatus Where Id = HotelOrder.StatusId) as Status From HotelOrder。
  2. 在应用程序中,此表经常作为DropDownList或者其他List控件的数据源。
  3. 这个表几乎从不改动。

2.数组及其问题

意识到这样设计存在问题,我们现在就想办法解决它。我们所想到的第一个办法是可以在程序中创建一个数组来表示预订状态,这样我们就可以删掉BookingStatus状态表(注意可以这样做是因为BookingStatus表的内容确定后几乎从不改动)。

string[] BookingStatus = {
"NoUse", "未提交","已提交","已取消","受理中","已退回","已订妥","已过期"
}; // 注意数组的0号元素仅仅是起一个占位作用,以使程序简洁。因为StatusId从1开始。

我们先看它解决了什么:上面提到的问题1、问题2都解决了,既不需要在数据库中创建表,又无需连接到数据库进行查询。

我们再看看当我们想要用文本显示酒店的预订时,该怎么做(假设有订单类HotelOrder,其属性StatusId代表订单状态,为int类型 )。

// GetItem用于获取一个酒店订单对象, orderId为int类型,代表订单的Id
HotelOrder myOrder = GetItem(orderId);
lbStatus.Text = BookingStatus[myOrder.StatusId]; //lbStatus是一个Label控件

目前为止看上去还不错,现在我们需要进行一个操作,将订单的状态改为“受理中”。

myOrder.StatusId = 4;

很不幸,我们发现了使用数组可能带来的第一个问题:不方便使用,当我们需要更新订单的状态值时,我们需要去查看BookingStatus数组的定义(除非你记住所有状态的数字值),然后根据状态值在数组中的位置来给对象的属性赋值。

我们再看另一个操作,如果某个订单的状态为“已过期”,就要对它进行删除:

if(BookingStatus[myOrder.StatusId]=="已过期"){
DeleteItem(myOrder); // 删除订单
}

此时的问题和上面的类似:我们需要手动输入字符串“已过期”,此时Vs2005 的智能提示发挥不了任何作用,如果我们不幸将状态值记错,或者手误打错,就将导致程序错误,较为稳妥的做法还是按下F12导向到BookingStatus数组的定义,然后将“已过期”复制过来。

现在,我们再看看如何来绑定到一个DropDownList下拉列表控件(Id为ddlStatus)上。

ddlStatus.DataSource = BookingStatus;
ddlStatus.DataBind();

但是我们发现产生的HTML代码是这样: