数往知来 三层架构 <十四>
三层架构_1
一、三层
就是把程序的各个部分都分离,尽量的底耦合,做到分工明确、责任明确
第一层:Dal 数据访问层
第二层 :Bll 业务逻辑判断层
第三层: UI 界面显示层
比如说数据访问的就专门负责执行对数据库的增删改查等操作,然后把查到的结果返回,其他的一概不管,
而逻辑判断层就专门负责对一些逻辑的判断,比如说用户登录给我一个用户名,然后我把用户名给数据访问层,让他到数据库里给我查有没有这个用户的信息,
然后把信息返回给我,然后我就进行盘断用户输入的用户名密码等是否正确,然后把登录的状态返回给UI层,
然后UI层就负责把用户名和密码传给逻辑判断层,让它进行判断用户名密码是否正确,然后告诉我登录的状态,ui层再给用户显示登录的状态,
这样做的好处:
--》代码的复用,就是说我要是用winfrom做一个程序然后用控制台再做一个程序,那么我逻辑判断层和数据访问层的代码值要写一次就可以了,以后直接拿来用就可以了
--》容易维护
三层做的登录小案例
================================数据访问层:对数据库进行操作=============================================
namespace _03三层实现登录
{
/// <summary>
/// 数据访问层
/// </summary>
public class Person_Dal
{
/// <summary>
/// 根据当前用户输入的用户名获取用户的详细信息
/// </summary>
/// <param name="uName"></param>
/// <returns></returns>
public Person LoginUID(string uName)
{
//1、SQL语句
string sql = "select * from Person where PLoginName=@name and PIsDel=0";
using (SqlDataReader read = SqlHelper.ExecuteReader(sql, System.Data.CommandType.Text, new SqlParameter("@name", uName)))
{
if (read.HasRows)
{
if (read.Read())
{
//把读出来的数据存到一个类对象的属性里面,这个类是公用的,返回这个对象,在其他的类也就可以访问这些数据了
Person per = new Person();
per.PCID = read.GetInt32();
per.PCName = read.GetString();
per.PIsDel = read.GetBoolean ();
per.PLoginName = read.GetString();
per.PPwd = read.GetString();
per.PType = read.GetInt32();
return per;
}
throw new Exception();
}
else
{
return null;
}
}
}
}
}
----- SqlHelper ---------
/// <summary>
/// 操作数据库的通用代码
/// 抽象类,不能实例化
/// </summary>
public abstract class SqlHelper
{
//APP配置文件的连接数据库的连接字符串
private static readonly string connstr = ConfigurationManager.ConnectionStrings["sql"].ConnectionString;
/// <summary>
/// 执行增删改操作,返回一个受影响行数
/// </summary>
/// <param name="cmdText">SQL语句</param>
/// <param name="cmdType">执行类型(存储过程、SQL语句等)</param>
/// <param name="par">以数组的形式提供执行命令的参数列表</param>
/// <returns></returns>
public static int ExecuteNonQuery(string cmdText, CommandType cmdType, params SqlParameter[] par)
{
using (SqlConnection con = new SqlConnection())
{
using (SqlCommand cmd=new SqlCommand ())
{
PreparaCommand(con, cmd, cmdText, cmdType, par);
return cmd.ExecuteNonQuery();
}
}
}
/// <summary>
/// 执行查询语句,返回第一行第一列
/// </summary>
/// <param name="cmdText"></param>
/// <param name="cmdType"></param>
/// <param name="par"></param>
/// <returns></returns>
public static object ExecuteScalar(string cmdText, CommandType cmdType, params SqlParameter[] par)
{
using (SqlConnection con = new SqlConnection())
{
using (SqlCommand cmd=new SqlCommand ())
{
PreparaCommand(con, cmd, cmdText, cmdType, par);
return cmd.ExecuteScalar();
}
}
}
/// <summary>
/// 返回一个SqlDataReader
/// </summary>
/// <param name="cmdText"></param>
/// <param name="cmdType"></param>
/// <param name="par"></param>
/// <returns></returns>
public static SqlDataReader ExecuteReader(string cmdText, CommandType cmdType, params SqlParameter[] par)
{
SqlConnection con = new SqlConnection();
using (SqlCommand cmd=new SqlCommand ())
{
PreparaCommand(con, cmd, cmdText, cmdType, par);
return cmd.ExecuteReader(System.Data .CommandBehavior .CloseConnection );
}
}
/// <summary>
/// 返回一个DataTable
/// </summary>
/// <param name="cmdText"></param>
/// <param name="cmdType"></param>
/// <param name="par"></param>
/// <returns></returns>
public static DataTable ExecuteDataTable(string cmdText, CommandType cmdType, params SqlParameter[] par)
{
using (SqlDataAdapter adapter=new SqlDataAdapter (cmdText ,connstr ))
{
DataTable dt = new DataTable();
if (par!=null)
{
adapter.SelectCommand.Parameters.AddRange(par);
}
adapter.SelectCommand.CommandType = cmdType;
adapter.Fill(dt);
return dt;
}
}
/// <summary>
/// 准备参数
/// </summary>
/// <param name="con">连接对象</param>
/// <param name="cmd">执行命令</param>
/// <param name="cmdText">SQL语句</param>
/// <param name="cmdType">执行命令的类型(存储过程、SQL语句等)</param>
/// <param name="par">以数组的形式提供执行命令的参数列表</param>
private static void PreparaCommand(SqlConnection con, SqlCommand cmd, string cmdText, CommandType cmdType, params SqlParameter[] par)
{
con.ConnectionString = connstr;
if (con.State == ConnectionState.Closed )
{
con.Open();
}
if (par!=null)
{
cmd.Parameters.AddRange(par);
}
cmd.Connection = con;
cmd.CommandText = cmdText;
cmd.CommandType = cmdType;
}
}
===============================================业务逻辑层:做一些逻辑判断========================================================
/// <summary>
/// 业务逻辑层
/// </summary>
public class Person_Bll
{
public Loginfet LoginSucceed(string uid, string pwd)
{
Person_Dal pd = new Person_Dal();
pd.LoginUID(uid);
//new一个Person对象,数据都存在person对象里
Person per = new Person();
string md5 = MD5Helper.MD5Change(pwd);
if (per==null)
{//如果是空的就是没有读取到数据
return Loginfet.LoginError ;
}
else if (per.PPwd !=md5 )
{//把密码转换成MD5值进行比较
return Loginfet.PasswordError ;
}
else
{
return Loginfet.OK ;
}
}
}
------------ 把登录状态写成枚举返回 ----------------
public enum Loginfet
{
OK,//登录成功
LoginError,//登录失败
PasswordError//用户名不存在
}
============================================================UI层 :向用户显示界面 ==================================
private void btnLogin_Click(object sender, EventArgs e)
{
Person_Bll pb = new Person_Bll();
Loginfet lf= pb.LoginSucceed(txtUID.Text.Trim(), txtPWD.Text.Trim());
if (lf== Loginfet.LoginError )
{
msgDiv1.MsgDivShow();
return;
}
else if (lf != Loginfet.PasswordError)
{
msgDiv1.MsgDivShow();
return;
}
else
{
msgDiv1.MsgDivShow("登录成功");
}
}
------------------------------------------------一个Person类 共有的 用来存储读取出来的用户的详细信息==========================================
public class Person
{
//PID, PCID, PType, PLoginName, PCName, PPYName, PPwd, PGender, PEmail, PAreas, PIsDel, PAddTime
public int PCID
{
get;
set;
}
public int PType
{
get;
set;
}
public string PLoginName
{
get;
set;
}
public string PCName
{
get;set;
}
public string PPwd
{
get;
set;
}
public bool PIsDel
{
get;
set;
}
}
MD5
一、MD5
MD5就是一整散列算法(Hash),任何长度的内容都可以计算出MD5散列值
--》MD5长度一般是32位的16进制数字符串(比如71f396e4134a1160d90bb1439876df31),MD5值的个数是有限的,但是源数据是无限的,
因此存在着不同的内容产生相同MD5值的概率。因此MD5算法不可逆,也就是只能得到内容对应的MD5值,无法由MD5值反推内容。
但是对不同的内容产生相同MD5值的概率非常非常非常低!同一个字符串计算出来的散列值是一样的
常用的地方
--》数据库存储的登录密码
数据库的保存的用户登录密码用明文来存储是不安全的,这样数据库管理员、黑客就很容易看到用户的密码了
Password字段保存用户输入密码的MD5值,这样系统管理员、黑客也不知道用户的密码是什么,也就避免了用户的其他系统密码被利用的问题判断密码正确性的方法:计算用户输入的密码的MD5值,与数据库存储的MD5值进行比较,如果相等则认为密码正确。
为什么很多网站只有密码重置,没有找回原密码功能?。
--》因为MD5具有不可逆性,数据库里存的 是MD5值,它也不知道你的密码是多少
--》MD5可以计算一个字符串的散列值,也可以计算一个文件的散列值
/// <summary>
/// 把字符串转换成MD5的方法
/// </summary>
/// <param name="str">要转换的字符串</param>
/// <returns></returns>
public string MD5Change(string str)
{
//需要一个MD5对象,MD5是密封类不能直接new
MD5 md5 = MD5.Create();
//把要转换的字符串转换成字节流byte[]
//把字符串转成byte数组的时候,使用不同编码转换结果有时会不同(一般指双字节字符。)
byte[] md5Byte = System.Text.Encoding.Default.GetBytes(str);
//根据这个字符串的字节数组计算出这个字符串的MD5值,返回的是一个byte数组
byte[] resMD5 = md5.ComputeHash(md5Byte );
//用完要清空
md5.Clear();
StringBuilder sb = new StringBuilder();
; i <resMD5 . Length; i++)
{ //直接tostring()返回的是十进制的,我们要的是十六进制
sb.Append(resMD5 [i].ToString ("x2"));
}
//把sb tostring()返回
return sb.ToString();
}
/// <summary>
/// 把文件转换成MD5
/// </summary>
/// <param name="path"></param>
/// <returns></returns>
public static string MD5ChangeFile(string path)
{
//把文件读进来
using (FileStream read=new FileStream (path,FileMode.Open ))
{
//创建一个MD5对象
MD5 md5 = MD5.Create();
//转换
byte[] MD5Byte= md5.ComputeHash(read);
md5.Clear();
StringBuilder sb = new StringBuilder();
; i <MD5Byte . Length; i++)
{
sb.Append(MD5Byte[i].ToString ("x2") );
}
return sb.ToString();
}
}
三层的作用
复习委托:把老师的MsgDiv插件看一看,里面是委托实现的
一、三层 为了实现程序模块化,把一些单独的工能单独的封装起来实现代码的分离,便于日后的维护,和代码的重用
--》表示层(UI)
作用:和用户交互,用户可以通过我们的程序看到数据库里的数据,展示数据给用户看,用户还可以提交数据
--》业务逻辑层(BLL)
作用:提供具体的业务逻辑,隔离UI和Dal
--》数据访问层(Dal)
作用:操作数据库
三个层之间的应用关系
数据访问层引用Model层
业务逻辑层应用数据访问层和Model层
表现层引用Model层和业务逻辑层
-->实体(Model)
作用:在三层中传递数据这个不属于三层里的一个层,是用来存放类和类之间的公共数据
数往知来 三层架构 <十四>的更多相关文章
- 深入浅出的webpack构建工具--webpack4+vue+router项目架构(十四)
阅读目录 一:vue-router是什么? 二:vue-router的实现原理 三:vue-router使用及代码配置 四:理解vue设置路由导航的两种方法. 五:理解动态路由和命名视图 六:理解嵌套 ...
- PowerBuilder编程新思维3:适配(三层架构与GraphQL)
PowerBuilder编程新思维3:适配(三层架构与GraphQL) PB在富客户端时代,是一线开发工具.随着网络发展,主流架构演进到三层架构的时代,PB拿不出有力的三层架构,已经明显力不从心,市场 ...
- Alink漫谈(十四) :多层感知机 之 总体架构
Alink漫谈(十四) :多层感知机 之 总体架构 目录 Alink漫谈(十四) :多层感知机 之 总体架构 0x00 摘要 0x01 背景概念 1.1 前馈神经网络 1.2 反向传播 1.3 代价函 ...
- winform学习日志(十九)----------真正三层架构之登录
摘要:一:三层构架的基础知识在项目开发的过程中,有时把整个项目分为三层架构,其中包括:表示层(UI).业务逻辑层(BLL)和数据访问层(DAL).三层的作用分别如下: 表示层:为用户提供交互操作界面, ...
- jquery 调用wcf 的SOA架构,将三层架构运用到SOA的架构中来(第四天)
经过前面3天的学习,我想大家应该对SOA的架构有了初步的了解,其实 SOA与三层架构并不冲突,而是三层架构的升级版. 来看下传统的三层架构! 一共可以分为4个层: 模型层(可有可无),客户端,服务端, ...
- C# Dapper基本三层架构使用 (四、Web UI层)
三层架构的好处,一套代码无论WinForm还是Web都可以通用,只写前台逻辑就可以了,现在展示Web调用三层的示例 首先在项目中创建一个Web MVC5项目,目前项目目录如下 在Web项目Web.co ...
- 第十四章——循环神经网络(Recurrent Neural Networks)(第二部分)
本章共两部分,这是第二部分: 第十四章--循环神经网络(Recurrent Neural Networks)(第一部分) 第十四章--循环神经网络(Recurrent Neural Networks) ...
- 我的MYSQL学习心得(十四) 备份和恢复
我的MYSQL学习心得(十四) 备份和恢复 我的MYSQL学习心得(一) 简单语法 我的MYSQL学习心得(二) 数据类型宽度 我的MYSQL学习心得(三) 查看字段长度 我的MYSQL学习心得(四) ...
- asp.net三层架构 及其中使用泛型获取实体数据介绍
asp.net中使用泛型获取实体数据可以发挥更高的效率,代码简洁方便,本例采用三层架构.首先在model层中定义StuInfo实体,然后在 DAL层的SQLHelper数据操作类中定义list< ...
随机推荐
- SSIS ->> Event Handler
Event Handler支持在某个事件触发的时候定义好处理该事件的逻辑,比如错误事件触发是该怎么处理.它跟Control Flow界面相似,就好像执行了另外一个包一样.Event Handler不仅 ...
- Maven+Spring+Hibernate+Shiro+Mysql简单的demo框架(一)
相关的maven的 pom.xml <project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="ht ...
- iPhone(iOS设备) 无法更新或恢复时, 如何进入恢复模式
在更新或恢复 iPhone 时,如果遇到以下所列问题之一.可能就要将设备置于恢复模式,并尝试重新恢复设备. 设备不断地重新启动,但从未显示主屏幕. 无法完成更新或恢复,且 iTunes 不再能识别设 ...
- [原]zoj3772--【水题】线段树区间查询+矩阵乘法
思路来源:http://blog.csdn.net/u013654696/article/details/23037407#comments [做浙大校赛的时候没有看这道题,事后做的.思路不是自己的, ...
- BZOJ 1009 GT考试(ac自动机+矩阵DP)
题目链接:http://61.187.179.132/JudgeOnline/problem.php?id=1009 题意:给定一个长度为m的串s.有多少种长度为n的串不包含s? 思路:(1)将s插入 ...
- MongoDB 学习笔记(一)基础篇
1.MongoDB 特点 面向集合存储,存储对象类型的数据方便 模式自由,不需要定义任何模式(schma) 动态查询 完全索引,包含内部对象 复制和故障恢复方便 高效的二进制数据存储 支持c# 平台驱 ...
- Android 工程在4.0基础上混淆
Android现在对安全方面要求比较高了,我今天要做的对apk进行混淆,用所有的第三方工具都不能反编译,作者的知识产权得到保障了,是不是碉堡了. 一,首先说明我这是在4.0基础上进行的. 先看看pro ...
- 《OD学hadoop》Linux基础
一.Linux基本环境 1. Linux常见版本及VMware虚拟机安装Linux系统 2. 虚拟机网络配置(IP地址.主机名.防火墙) 3. 文件基本命令操作 4. 四大远程连接工具使用 二.Lin ...
- 维护没有源代码,float改成double
float f= 931340.31f; Console.WriteLine(f.ToString("#,###,##0.00")); 返回 931,340.30 ,float 1 ...
- volley(4) 请求参数:data:[ { bar_remain:XX , bar_code:"XX" , bar_id: XX}], method:GET
1. 来自于WHCombineBatchFragment.java 2.部分代码 ).).).).port + Url.LABELPRINT + "?data="+strPrint ...