对于机房收费系统的重构,从大的方面来看,无非就是对于数据库的四个操作,增删改查。并且我们用的是三层架构进行重构,D层用来和数据库打交道,进行这四个操作就需要有返回值,增删改在这里不多说,但是当进行查的时候,d层需要数据给b层,那么我们以什么样的形式返回最好呢?
刚开始接触三层的时候,我返回的是datatable,就是dt.Rows[0][“xxx”]的形式,感觉实现了,就很厉害了,后来通过与大家交流发现,其实还有一种更好用的方法,就是利用实体类填充泛型集合。
那么问题就出来了:
泛型,顾名思义就是泛泛的类型. 也就是没有确定的类型. 那么没有确定类型怎么使用呢?实际上,使用的时候规定类型就行了.
集合,就是一种处理多个数据类型的类,并且一般你会在多个应用程序中使用同一个集合的多种不同的形式。你不需要每次根据草稿建立集合,而是使用泛型建立一个泛型类原型(prototype)。
在使用的时候,根据需要处理的数据类型,将List<T>尖括号中的T换成对应的类型,并创建对应的实例就可以使用了.
首先,你需要给代码添加如下所示的Imports语句:
Imports System.Collections.Generic |
添加Imports语句之后,你就可以建立泛型类了。基本的类看起来很常见。你可以使用属性、函数、子程序、字段或可以在类中使用的其它任何东西。
Public Class ConvertHelperEntity Public Shared Function convertToList(Of T As {New})(ByVal dt As DataTable) As IList(Of T) '将datatable 转换成泛型集合 Dim myList As New List(Of T) '定义最终返回的集合 Dim myType As Type = GetType(T) '得到实体类的类型名 Dim dr As DataRow '定义行集 Dim tempName As String = String.Empty '定义一个临时变量 '遍历datatable所有数据行 For Each dr In dt.Rows Dim myT As New T '定义一个实体类的对象 Dim propertys() As PropertyInfo = myT.GetType.GetProperties() '定义属性集合 Dim pr As PropertyInfo '遍历该对象的所有属性 For Each pr In propertys tempName = pr.Name '将属性名称赋给临时变量 '检查datatable 是否包含此列(列名==对象的属性名) If (dt.Columns.Contains(tempName)) Then '将此属性与datatable里列名相比较,查看datatable是否包含此属性 '判断此属性是否有setter If (pr.CanWrite = False) Then '判断此属性是否可写,如果不可写,则跳出循环 Continue For End If Dim value As Object = dr(tempName) '定义一个对象型的变量来保存列的值 If (value.ToString <> DBNull.Value.ToString()) Then '如果非空,则赋给对象的属性 pr.SetValue(myT, value, Nothing) '在运行期间,通过反射,动态的访问一个对象的属性 End If End If Next myList.Add(myT) '添加到集合 Next Return myList '返回实体集合 End Function End Class我们需要在实体层,添加一个实体类, 因为创建集合的基础是要有类,然后才能将对象放入集合中。
那么我们为什么要说它比datatable要好呢?用泛型集合,我们在D层把DataTable转换成单个实体类,再把实体类填充到泛型集合中。
其核心思想图:
Datatable |
List<T> |
非常容易写错,编译器不检查 |
按一下点,自己出来,不会写错 |
必须了解数据库的结构 |
不必了解数据库结构 |
不符合面向对象思想 |
符合面向对象思想 |
DataTable为弱类型,无法直观的看出字段的数据类型。 |
实体类的属性是强类型,每个字段的类型都是已知的。 |