【C#】七层登录——机房重构
三层
一、数据访问层(DAL):使用ADO.NET中的数据操作类,为数据库中的每个表,设计1个数据访问类。类中实现:记录的插入、删除、单条记录的查询、记录集的查询、单条记录的有无判断等基本的数据操作方法。对于一般的管理信息软件,此层的设计是类似的,包含的方法也基本相同。此层的任务是:封装每个数据表的基本记录操作,为实现业务逻辑提供数据库访问基础。
二、业务逻辑层(BLL):为用户的每个功能模块,设计1个业务逻辑类,此时,需要利用相关的数据访问层类中,记录操作方法的特定集合,来实现每个逻辑功能。
三、界面层(UI):根据用户的具体需求,为每个功能模块,部署输入控件、操作控件和输出控件,并调用业务逻辑层中类的方法实现功能。
层之间的调用关系
数据访问层的类,直接访问数据库,实现基本记录操作。
业务逻辑层的类,调用相关的数据访问类,实现用户所需功能。
界面层:部署控件后,调用业务逻辑层的类,实现功能。
将应用程序的功能分层后,对于固定的DBMS,数据访问层基本可以不变,一旦用户的需求改变,首先修改业务逻辑层,界面层稍做改动即可。这种做法使程序的可复用性、可修改性,都得到了很好的改善,大大提高了软件工程的效率。
七层
一、实体层(Entity):实体层,存放全局的实体类,方便各个层之前的参数调用。
二、外观层(Facade):这里用到了外观模式,使得复杂问题简单化,用外观模式定义系统中每一层的入口,层与层之间不直接产生联系,而通过外观类建立联系,降低层之间的耦合度。
三、工厂层(Factory):工厂来创建接口,返回接口,用到了抽象工厂+反射+配置文件,作用是灵活的实现数据库的连接,方便换数据库,进一步解耦合。
四、接口层(IDAL):接口层用来定义一个统一的接口,解除B层和D层的耦合。
七层是在三层的基础上与设计模式相结合演化而来的,应用了设计模式中的外观模式和抽象工厂模式。
其实还有一个SQLHelper,因为SQLHelper是从DAL层分离出来了,不打算做一层,否则就是8层了。
包图
实现步骤
1.创建Entity,实现业务实体,存放全局的实体类,方便各层之间的参数调用。
2.创建IDAL,实现接口,定义一个统一的接口。
3.创建DAL,实现接口里的方法,和数据库直接交互。
4.添加APP.config里的配置信息,为DAL提供程序集。
5.创建Factory,返回程序集合的指定类的实现。
6.创建BLL,调用Factory,得到程序集指定类的实例,完成数据操作方法。
7.创建Facade,调用BLL,得到BLL层的处理结果返回值。
8.创建UI,调用Facade里的数据操作方法,实现登录。
代码实现
1.Entity层
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
namespace Entity
{
public class UserEntity
{
//定义 用户ID 字段
private int? userid;
public int? UserID
{
get { return userid; }
set { userid = value; }
}
//定义 用户名 字段
private string userName;
public string UserName
{
get { return userName; }
set { userName = value; }
}
//定 密码 字段
private string pwd;
public string PWD
{
get { return pwd; }
set { pwd = value; }
}
//定义 等级 字段
private string level;
public string Level
{
get { return level; }
set { level = value; }
}
//定义 状态 字段
private bool stat;
public bool Stat
{
get { return stat; }
set { stat = value; }
}
}
}
2.IDAL
namespace IDAL
{
public interface IUser
{
//检查用户是否存在/密码是否正确
Boolean SelectExistUser(UserEntity enUser);
}
}
3.DAL
namespace DAL
{
public class UserDAL : IUser
{
public Boolean SelectExistUser(UserEntity enUser)
{
//连接数据库
string ConStr = "server=.; database=JFCG_LMJ; uid = sa ; pwd=123456";
SqlConnection conn = new SqlConnection(ConStr);
//采用参数化查询,防止SQL注入的发生
SQLHelper sqlHelper = new SQLHelper();
SqlParameter[] sqlParams = {
new SqlParameter("@userID", enUser.UserID),
new SqlParameter("@PWD", enUser.PWD)
};
//将sql语句写在存储过程“procUserSelectAdminVaild”中,然后引用存储过程
string str = "procUserSelectAdminVaild";
SqlCommand com = new SqlCommand(str, conn);
conn.Open();
com.CommandType = CommandType.StoredProcedure ;
com.Parameters.AddRange(sqlParams);
//在 User_Info 数据表中查找是否有当前登录用户
SqlDataReader sdr = com.ExecuteReader();
Boolean table = sdr.Read();//用 Read 方法读取数据
return table;
}
}
}
4.APP.config里的配置信息
<appSettings>
<add key ="ConnStr" value="server=.; database=JFCG_LMJ; uid = sa ; pwd=123456"/>
<!-- server是自己数据库的名字或者用.代表本地;把database,uid,pwd修改为与自己数据库对应的关系-->
<add key ="DB" value="DAL" />
</appSettings>
5.Factory
namespace Factory
{
public class UserFactory
{
//接收来自配置文件的数据
string StrDB = System.Configuration.ConfigurationManager.AppSettings["DB"];
public IUser CreateUserInfo()
{
string ClassName = StrDB + "." + "UserDAL";//DAL层的类名
// ILoginIDAL Iuser= (ILoginIDAL)Assembly.Load(StrDB).CreateInstance(ClassName);
return (IUser)Assembly.Load(StrDB).CreateInstance(ClassName); //反射加工厂的应用
}
}
}
6.BLL
namespace BLL
{
public class UserBLL
{
public bool CheckUser(UserEntity enUser)
{
//检查用户是否存在
Factory.UserFactory Fact = new Factory.UserFactory();//实例化工厂
IDAL.IUser IUser = Fact.CreateUserInfo();//调用工厂方法创建接口
Boolean flag = IUser.SelectExistUser(enUser);//接受D层的返回值
return flag;
}
//判断用户登录密码是否正确
public bool CheckPWD(UserEntity enUser)
{
//检查用户是否存在
Factory.UserFactory fact = new Factory.UserFactory();//实例化工厂
IDAL.IUser Iuser = fact.CreateUserInfo();//调用工厂方法创建接口
Boolean table = Iuser.SelectExistUser(enUser);//接受D层的返回值
return table;
}
}
}
7.Facade
namespace Facade
{
public class UserFacade
{
//检查用户是否存在
public Boolean CheckUser(UserEntity enUser)
{
bool flag;
BLL.UserBLL IsUserExist = new BLL.UserBLL();
flag = IsUserExist.CheckUser(enUser);
return flag;
}
//检查密码是否正确
public Boolean CheckPWD(UserEntity enUser)
{
bool table;
BLL.UserBLL IsPwdExist = new BLL.UserBLL();
table = IsPwdExist.CheckPWD(enUser);
return table;
}
}
}
8.UI
namespace UI
{
public partial class UILogin : Form
{
public UILogin()
{
InitializeComponent();
}
private void BtnLogin_Click(object sender, EventArgs e)
{
//判断输入不能为空
if (txtUserID .Text .Trim()=="")
{
MessageBox.Show("请输入用户名!", "登录", MessageBoxButtons.OK, MessageBoxIcon.Warning);
txtUserID.Focus();
}
if (txtPWD.Text == "")
{
MessageBox.Show("请输入密码!", "登录", MessageBoxButtons.OK, MessageBoxIcon.Warning);
txtPWD.Focus();
}
//try
//{
//Facade.LoginFacade Facade = new Facade.LoginFacade();
Entity.UserEntity enUser = new Entity.UserEntity
{
UserID = Convert.ToInt32(txtUserID.Text.Trim()),
PWD = txtPWD.Text.Trim()
};
Boolean flag = false;
Facade.UserFacade facadeLogin = new Facade.UserFacade();//实例化外观
flag = facadeLogin.CheckUser(enUser);//调用外观的方法,返回给user
if (flag != false)
{
this.Hide(); //隐藏当前窗体
MessageBox.Show("登录成功!");
//在这里添加一个用户登录记录
frmMain frmMain = new frmMain();//实例化一个窗体
frmMain.Show(); //显示实例化的窗体
}
else
{
MessageBox.Show("密码或者用户名错误");
txtUserID.Text = "";
txtPWD.Text = "";
txtUserID.Select();
txtUserID.Focus();
}
}
// catch (Exception)
// {
// //throw;
// }
//}
private void BtnClose_Click(object sender, EventArgs e)
{
Environment.Exit(0);
}
}
}
9.SQLHelper
using System.Data;
using System.Data.SqlClient;
using System.Configuration;
namespace DAL
{
public class SQLHelper
{
private SqlConnection conn = null;
private SqlCommand cmd = null;
private SqlDataReader sdr = null;
public SQLHelper()
{
string connStr = ConfigurationManager.AppSettings["ConnStr"];
conn = new SqlConnection(connStr);
}
private SqlConnection GetConn()
{
if (conn.State == ConnectionState.Closed)
{
conn.Open();
}
return conn;
}
/// <summary>
/// 执行不带参数的的增删改SQL语句或者存储过程
/// </summary>
/// <param name="cmdText">增删改查SQL</param>
/// <param name="ct">命令类型</param>
/// <returns>返回受影响的行数</returns>
public int ExecuteNonQuery(string cmdText, CommandType ct)
{
int res;
try
{
cmd = new SqlCommand(cmdText, GetConn());
cmd.CommandType = ct;
res = cmd.ExecuteNonQuery();
}
catch (Exception ex)
{
throw ex;
}
finally
{
if (conn.State == ConnectionState.Open)
{
conn.Close();
}
}
return res;
}
/// <summary>
/// 执行带参数的的增删改SQL语句或者存储过
/// </summary>
/// <param name="cmdText">增删改查SQL</param>
/// <param name="paras">要查询的参数</param>
/// <param name="ct">命令类型</param>
/// <returns>返回受影响的行数</returns>
public int ExecuteNonQuery(string cmdText, SqlParameter[] paras, CommandType ct)
{
int res;
using (cmd = new SqlCommand(cmdText, GetConn()))
{
cmd.CommandType = ct;
cmd.Parameters.AddRange(paras);
res = cmd.ExecuteNonQuery();
}
return res;
}
/// <summary>
/// 执行不带参数的查询SQL语句或存储过程
/// </summary>
/// <param name="cmdText">查询SQL语句或存储过程</param>
/// <param name="ct">命令类型</param>
/// <returns></returns>
public DataTable ExecuteQuery(string cmdText, CommandType ct)
{
DataTable dt = new DataTable();
cmd = new SqlCommand(cmdText, GetConn());
cmd.CommandType = ct;
using (sdr = cmd.ExecuteReader(CommandBehavior.CloseConnection))
{
dt.Load(sdr);
}
return dt;
}
/// <summary>
/// 执行带参数的查询SQL语句或存储过程
/// </summary>
/// <param name="cmdText">查询SQL语句或存储过程</param>
/// <param name="paras">参数集合</param>
/// <param name="ct">命令类型</param>
/// <returns></returns>
public DataTable ExecuteQuery(string cmdText, SqlParameter[] paras, CommandType ct)
{
DataTable dt = new DataTable();
cmd = new SqlCommand(cmdText, GetConn())
{
CommandType = ct
};
cmd.Parameters.AddRange(paras);
using (sdr = cmd.ExecuteReader(CommandBehavior.CloseConnection))
{
dt.Load(sdr);
}
return dt;
}
}
}