机房收费系统开始了有一段时间了,但是感觉自己对于这一条线还是理解的不太深刻,导致后面也快不起来,所以来分析一下这用户登录的过程。
如果是纯三层的话,理解起来差不多,无非就是在层与层之间传递,但是随着用户需求,软件功能越来越多,那么其复杂程度越来越大。
因此,我们引入了更多层,从图中看,我们觉得好像这样比三层更加复杂,但是实现过程中我们会体现其高内聚,低耦合的优点。我们不难发现这是从三层加入设计模式演化而来,之所以采用抽象工厂模式是考虑到换数据库的方便,而应用外观模式是为了解决UI层和BLL层耦合性过高的问题,而UI层不用知道BLL层的存在,而Facade知道BLL层哪些类负责哪些请求,将UI层的请求代理给适当的BLL层的类。
整体有了,接下来就是细节了,也就是从宏观到微观。
第一步要做的就是抽象实体层的类(Entity),因为信息系统是对数据的操作和处理,首先
必须要有数据,这个时候,我们要返回需求,了解用户的数据要求,以此为依据进行数据库设计。
在这里我们就需要抽象出UserEntity类。
<span style="font-family:KaiTi_GB2312;font-size:18px;">Public Class UserEntity Private _userName As String Private _PWD As String Private _userLevel As String Private _accounter As String Public Shared UserHead As String Public Property userName() As String Get Return _userName End Get Set(value As String) _userName = value End Set End Property Public Property PWD() As String Get Return _PWD End Get Set(value As String) _PWD = value End Set End Property Public Property userLevel() As String Get Return _userLevel End Get Set(value As String) _userLevel = value End Set End Property Public Property accounter() As String Get Return _accounter End Get Set(value As String) _accounter = value End Set End Property End Class</span>
<span style="font-family:KaiTi_GB2312;font-size:18px;">Imports BLL Imports Entity Public Class frmLoginUI Private Sub btOK_Click(sender As Object, e As EventArgs) Handles btOK.Click Dim enUser As New Entity.UserEntity 'Dim loginFA As Facade.loginFacade '实例化过程 enUser.userName = txtUser.Text.Trim '赋值过程 enUser.PWD = txtPWD.Text.Trim 'Dim strResult As String '判断是否输入了用户名及密码 If txtUser.Text.Trim() = "" Then MessageBox.Show("请输入用户ID", "", MessageBoxButtons.OK, MessageBoxIcon.Exclamation) txtUser.Select() txtUser.Focus() Exit Sub ElseIf IsNumeric(txtUser.Text) = False Then MsgBox("用户名请输入数字") txtUser.Text = "" txtUser.Select() txtUser.Focus() Exit Sub ElseIf txtPWD.Text.Trim() = "" Then MessageBox.Show("请输入密码", "", MessageBoxButtons.OK, MessageBoxIcon.Exclamation) txtPWD.Select() txtPWD.Focus() Exit Sub End If 'Try '定义一个外观层的对象 Dim FacadeLogin As New Facade.loginFacade Dim strResult1 As Boolean strResult1 = FacadeLogin.CheckUser(enUser) If strResult1 = False Then MsgBox("用户不存在") txtUser.Text = "" txtUser.Select() txtUser.Focus() End If Dim table As DataTable table = FacadeLogin.CheckPwd(enUser) If Trim(txtPWD.Text) = Trim(table.Rows(0).Item(1)) Then MsgBox("登陆成功") Me.Hide() frmMainUI.Show() Else MsgBox("密码不正确") txtPWD.Text = "" txtPWD.Select() txtPWD.Focus() End If End Sub Private Sub btCancel_Click(sender As Object, e As EventArgs) Handles btCancel.Click End End Sub Private Sub frmLoginUI_Load(sender As Object, e As EventArgs) Handles MyBase.Load txtUser.Select() txtUser.Focus() End Sub End Class</span>
外观其实就是方法的封装,而对于外观层在登陆的时候其实作用不大,但同样调用B层方法。
<span style="font-family:KaiTi_GB2312;font-size:18px;">Imports BLL Public Class loginFacade Public Function CheckUser(ByVal enUser As Entity.UserEntity) As Boolean '用于检查用户是否存在 Dim IsUserExistsBLL As New BLL.IsExists Dim flag As Boolean flag = IsUserExistsBLL.IsUserExists(enUser) If flag = True Then Return True Else Return False End If End Function '检查用户是否正确 Public Function CheckPwd(ByVal enUser As Entity.UserEntity) As DataTable Dim IsPwdBLL As New BLL.IsExists Dim table As DataTable table = IsPwdBLL.CheckUserPWD(enUser) Return table End Function End Class</span>
<span style="font-family:KaiTi_GB2312;font-size:18px;">Imports IDAL Imports Entity Public Class IsExists '检查用户是否存在 Public Function IsUserExists(ByVal enUser As Entity.UserEntity) As Boolean '定义并实例化一个工厂 Dim factory As New Factory.DataAccess '定义一个接口变量 Dim IUser As IUser '通过调用创建用户的工厂方法 IUser = factory.CreateUser Dim table As DataTable Dim flag As Boolean table = IUser.CheckExistsUser(enUser) If table.Rows.Count = 0 Then flag = False Else flag = True End If Return flag End Function '查密码是否正确 Public Function CheckUserPWD(ByVal enUser As UserEntity) As DataTable Dim factory As New Factory.DataAccess '定义工厂并实例化工厂变量factory Dim IUser As IUser '定义接口变量IUser Dim table As DataTable '中间变量,用于存储D层查询到的数据 IUser = factory.CreateUser '调用工厂的CreateUser方法创建iuser接口实例 table = IUser.CheckExistsUser(enUser) '调用接口的方法UserIsExist,并将返回值返回给flag Return table End Function End Class</span>
经典的工厂层的设计,用反射+配置文件,防止更换数据库。
<span style="font-family:KaiTi_GB2312;font-size:18px;">Imports System.Reflection Imports System.Configuration Imports IDAL '/**************************************************************************** '类 名 称:SqlServerFactory '命名空间:Factory '内 容:读配置文件来给DB字符串赋值,在配置文件中写明是SqlServer还是Access,想要更换数据 '库直接改配置文件就行。前提是:Sqlserver前缀的DAL层类里写的是访问SQLServer数据库的代码。 'Access前缀的DAL层类里写的是访问Access数据库的代码。 '功 能:用反射+配置文件+抽象工厂,方便更换数据库 '创建时间:2014/8/14 10:40:59 '作 者:王金博 '修改时间: '修 改 人: '版 本 号:v1.0.0 '****************************************************************************/ Public Class DataAccess '利用反射+配置文件+抽象工厂 Private Shared ReadOnly AssemblyName As String = "DAL" '定义程序集名称变量,D层命名空间的名字 Private Shared ReadOnly db As String = ConfigurationManager.AppSettings("DB") '表示读配置文件,如果配置文件中是Sqlserver,就访问SQLServer数据库,如果是别的就访问别的,不用更改程序中的代码。 '创建用户表的工厂 Public Function CreateUser() As IUser Dim className As String = AssemblyName + "." + db + "UserDAL" 'AssemblyName是程序集的名称,db+"UserDAL"是DAL层中的SqlServerUserDAL。如果不用SqlServer数据库,那么我在 'Factory中再另建一个类,比如访问Access数据库,那么类名就叫AccessUserDAL。把配置文件中Value 值改为 'Access。这样就是扩展而不是修改。 Dim iuser As IUser iuser = CType(Assembly.Load(AssemblyName).CreateInstance(className), IUser) '将实例化的D层通过向上转型转换成接口类,然后通过调用接口类中的函数来调用D层中实现该接口的函数 Return iuser End Function End Class</span>
定义一个用于解耦B层和D层的接口
<span style="font-family:KaiTi_GB2312;font-size:18px;">Public Interface IUser '检查用户是否存在和密码是否正确 Function CheckExistsUser(ByVal enUser As Entity.UserEntity) As DataTable End Interface</span>
用于连接数据库,实现接口
<span style="font-family:KaiTi_GB2312;font-size:18px;">Imports Entity Imports IDAL Imports System.Data.SqlClient Public Class SqlserverUserDAL : Implements IUser Public Function SelectExistsUser(enUser As UserEntity) As DataTable Implements IUser.CheckExistsUser Dim sql As String '定义字符串变量sql,用于存放要执行的sql语句 Dim table As DataTable '定义表变量table,用于存储执行的结果并返回 Dim paras As SqlParameter() = {New SqlParameter("@userName", enUser.userName), New SqlParameter("@PWD", enUser.PWD)} sql = "select * from T_User where userName=@UserName and PWD=@PWD" '存储sql语句 table = SqlHelper.SqlHelper.GetDataTable(sql, CommandType.Text, paras) '执行sql语句,将结果赋给table Return table End Function End Class </span>总之,我们尽量做到符合单一职责原则,一个类完成一个功能,高内聚,低耦合,使得每层之间联系变得很少,而使用设计模式让我们的思路更加清晰,能够让系统更好地去扩展以及维护。