一、普通架构中存在的问题

StudentDB数据库,包含一张StudentInfoTB表,结构如下:

s_id int primary key identity(1,1),
s_name Nvarchar(10) not null,
s_age int check(s_age >10 and s_age<30),
s_sex bit not null

先来看一下普通的架构的问题所在:

调用关系:

dal层代码只是通过SqlHelper简单的操作一下数据库,就不展示了。

StudentInfo的bll层代码,实例化了一个dal层对象,并且每个方法返回对应的方法:

private StudentInfoADODal dal = new StudentInfoADODal();
public List<StudentInfoModel> Select()
{
return dal.Select();
} public int Update(StudentInfoModel siModel)
{
return dal.Update(siModel);
} public int Delete(int id)
{
return dal.Delete(id);
} public int Add(StudentInfoModel siModel)
{
return dal.Add(siModel);
} public StudentInfoModel Get(int id)
{
return dal.Get(id);
}

ui层调用:

static void Main(string[] args)
{
Display();
Console.ReadKey();
} static void Display()
{
//实例化bll层对象
StudentInfoBll bll = new StudentInfoBll();
//接收返回值
List<StudentInfoModel> lstModel = bll.Select();
//输出标题
Console.WriteLine($"编号\t姓名\t年龄\t性别");
//循环输出
foreach (StudentInfoModel item in lstModel)
{
Console.WriteLine($"{item.s_id}\t{item.s_name}\t{item.s_age}\t{item.s_sex}");
}
}

输出:

一切看起来都没什么问题,可是如果后续逻辑复杂起来了,并且Dal层对象有变动的话,我们是不是需要在Bll层修改实例化的Dal层对象,而且一但更改,所有位置的Dal层对象都会修改,现在代码很少,东西也不复杂,可是如果代码复杂了,修改的话就不是太方便了。

首先我们应该想到的是工厂模式,因为如果Dal层有变动的话,我们只需要修改工厂模式中返回的对象就行了,但是如果使用简单工厂模式的话,返回值是个问题,Dal层改变的话,返回值肯定也会发生改变,到时候还是要对bll层接收对象修改类型。

二、抽象工厂+IDal

所以我们应该考虑使用抽象工厂,既然要用抽象工厂,那么就必须有一个具体的父类或者接口,但是Dal层并没有继承任何父类或实现任何接口(Object除外),所以我们应该抽象出一个IDal层,使得所有的Dal层都要去实现IDal层,然后用IDal层的接口作为返回值,返回给Bll层,后续如果在要修改Dal层的话,只要Dal层实现了IDal层我们是不是就不用在对Bll层做任何修改了。

接下来先创建IDal层:

然后抽象出IStudentInfoDal接口:

public interface IStudentInfoDal
{
List<StudentInfoModel> Select();
int Update(StudentInfoModel siModel);
int Delete(int id);
int Add(StudentInfoModel siModel);
StudentInfoModel Get(int id);
}

StudentInfoADODal层实现IStudentDal接口:

下面就是使用工厂模式来进行创建对象,先创建工厂层和DalFactory类:

public class DalFactory
{
public static IStudentInfoDal CreateStudentInfoInstance()
{
return new StudentInfoADODal();
}
}

Bll层调用的话直接声明接口,然后通过工厂模式来获取Dal层对象:

private IStudentInfoDal dal = DalFactory.CreateStudentInfoInstance();

这下如果后续需要更改的话,直接就更改工厂模式这一个地方就行了,其他地方就不用做更改了。

现在看一下调用关系,箭头是引用关系

可以发现,Dal层并没有做到完全独立起来,Factory层还是在引用Dal层,更换数据库的话,还是要重新添加引用,重新修改工厂模式中的代码。小项目不分层都是可以的,如果项目很复杂,做任何事情之前都要考虑周到,任何细节都要处理好,所以就想办法把Dal层完全独立起来

三、反射+App.config实现抽象工厂

反射就是能够动态加载程序集,不需要添加对程序集的引用,就可以获取程序集内部的结构(属性、方法),可以实现动态创建对象,调用对象的方法,为属性赋值等操作。所以,在创建Dal层对象时,我们可以考虑使用反射来创建。

反射所在命名空间:System.Reflection;  其实就是先将dll文件给加载到Assembly对象中,然后通过Assembly对象创建dll文件中的对象(反射还有其他的几个常用的对象Type、Activator、PropertyInfo...)

使用反射创建dal层对象:

public static IStudentInfoDal CreateStudentInfoInstance()
{
//使用Assembly来加载程序集
Assembly assembly = Assembly.Load("CKKA.ADODal");
//通过assembly对象来创建一个StudentInfoADODal实例
//必须是完整的类型名称 类型所在命名空间+“.”+类名
Object siDal = assembly.CreateInstance("CKKA.ADODal.StudentInfoADODal");
return siDal as IStudentInfoDal;
}

然后右键CKKA.ADODal-->属性-->最左侧生成-->下方输出路径改为Ui/bin/debug或者Ui/bin(具体可以自己打开文件夹下看那个目录下有dll文件),详情看这篇帖子https://www.cnblogs.com/ckka/p/11331037.html

现在运行该程序是可以反射成功的

我们把将抽象工厂再次改造一下,将CKKA.ADODal和后缀ADODal写在App.config的appSettings节点下(web项目写在web.config)

<add key="AssemblyName" value="CKKA.ADODal"/>
<add key="Suffix" value="ADODal"/>

抽象工厂改进为(需要为Factory添加System.Configuration程序集和命名空间的引用):

public class DalFactory
{
//从配置文件中读取AssemblyName(程序集名称)和Suffix(Dal层扩展名)
private static String AssemblyName = ConfigurationManager.AppSettings["AssemblyName"];
private static String Suffix = ConfigurationManager.AppSettings["Suffix"]; //每个方法都调用此方法来创建对象
private static Object CreateInstance(String TypeName)
{
return Assembly.Load(AssemblyName).CreateInstance(TypeName);
} public static IStudentInfoDal CreateStudentInfoInstance()
{
//拼接类型名称
String TypeName = $"{AssemblyName}.StudentInfo{Suffix}";
//创建实例
return CreateInstance(TypeName) as IStudentInfoDal;
} }

以后我们如果需要更换数据库的话,只需要修改配置文件,操作数据库的Dal实现IDal接口,并且bin目录下有dll文件就行了,不需要更改任何代码。

由于没有其他数据库,我们就用EF操作数据库来测试一下:

将CKKA.EFDal的生成路径更改一下,然后修改配置文件为:

然后运行:

可以发现,完全不用改任何代码,就可以做到更换一整个Dal或者数据库,最终调用结构为:

如果我哪里写的有问题或者我说的不够清楚或者你有疑问的话,欢迎留言

简单架构:反射实现抽象工厂+IDAL接口完全独立DAL的更多相关文章

  1. 泛型、反射和抽象工厂结合解决多DB问题

  2. Asp.Net 设计模式 之 “工厂方法”即利用 接口 实现的抽象工厂

    主要改动部分: /// <summary>    /// 6.创建工厂方法模式(抽象工厂:接口)    /// </summary>    interface IFactory ...

  3. java 抽象工厂模式简单实例

    抽象工厂模式:提供一个创建一系列的相关的或者依赖的对象的接口,无需指定它们的具体实现类,具体的时间分别在子类工厂中产生. 类似于工厂模式:隔离了具体类的生产实现,使得替换具体的工厂实现类很容易.包含有 ...

  4. Head First设计模式——简单工厂、工厂、抽象工厂

    前言:按照惯例我以Head First设计模式的工厂模式例子开始编码学习.并由简单工厂,工厂模式,抽象工厂模式依次演变,归纳他们的相同与不同. 话说Head First认为简单工厂并不是设计模式,而是 ...

  5. python 设计模式之工厂模式 Factory Pattern (简单工厂模式,工厂方法模式,抽象工厂模式)

    十一回了趟老家,十一前工作一大堆忙成了狗,十一回来后又积累了一大堆又 忙成了狗,今天刚好抽了一点空开始写工厂方法模式 我看了<Head First 设计模式>P109--P133 这25页 ...

  6. 设计模式(3)--抽象工厂模式(Absrtact Factory Pattern)

    定义 抽象工厂模式的实质就是提供接口来创建一系列相关或独立的对象而不指定这些对象的具体类. 理解 在软件系统中,经常面临着"一系列相互依赖的对象"的创建工作:同时由于需求的变化,往 ...

  7. 大话设计模式C++版——抽象工厂模式

    前面说过,简单工厂模式是最基础的一种设计模式,那以工厂命名的设计模式就是23种设计模式中最多的一种,他们一脉相承,一步一步进化而来,这里就是其中的最后一种——抽象工厂模式(Abstract Facto ...

  8. 深入浅出设计模式——抽象工厂模式(Abstract Factory)

    模式动机在工厂方法模式中具体工厂负责生产具体的产品,每一个具体工厂对应一种具体产品,工厂方法也具有唯一性,一般情况下,一个具体工厂中只有一个工厂方法或者一组重载的工厂方法.但是有时候我们需要一个工厂可 ...

  9. C#学习笔记-抽象工厂模式

    题目1:数据访问,通过数据库对用户表单的进行访问,数据库包含SQL Server,对用户表单进行“新增用户”和“查询用户”信息等操作. 分析: 首先,确认用户表单,里面包含两个ID和Name两个字段, ...

随机推荐

  1. [leetcode] #213 House Robber II Medium (medium)

    原题链接 比子母题House Robber多了一个条件:偷了0以后,第n-1间房子不能偷. 转换思路为求偷盗[0,n-1)之间,以及[1,n)之间的最大值. 用两个DP,分别保存偷不偷第0间房的情况. ...

  2. [leetcode]95 Unique Binary Search Trees II (Medium)

    原题 字母题添加链接描述 一开始完全没有思路.. 百度看了别人的思路,对于这种递归构造的题目还是不熟,得多做做了. 这个题目难在构造出来.一般构造树都需要递归. 从1–n中任意选择一个数当做根节点,所 ...

  3. sql注入篇1

    一.前言 学习了感觉很久的渗透,总结一下sql注入,系统整理一下sql注入思路. 二.关于sql注入 所谓SQL注入,就是通过把SQL命令插入到Web表单提交或输入域名或页面请求的查询字符串,最终达到 ...

  4. MySQL 5.7和8.0性能测试

    目录 背景 前提 环境 测试 双1模式下 0 2 模式下 结论 背景 测试mysql5.7和mysql8.0 分别在读写.只读.只写模式下不同并发时的性能(tps,qps) 前提 测试使用版本为mys ...

  5. python课堂整理16---内置函数

    1. abs :求绝对值 print(abs(-1)) 2. all()传入一个可迭代对象,对该对象进行bool值运算,若都为True 就返回True,有一个为假,就返回False print(all ...

  6. Python基础总结之第五天开始【认识简单的条件语句,也可以叫判断语句】(新手可相互督促)

    周五更新很累... 坚持,年薪20万又进了一步~~ python中的条件语句以[ if ]开头,条件语句成立时,运行该代码块,如果条件不成立,则跳过该代码块,执行后面的代码块. 简单的小示例: 输入性 ...

  7. 用wxpy管理微信公众号,并利用微信获取自己的开源数据。

    之前了解到itchat 乃至于 wxpy时 是利用tuling聊天机器人的接口.调用接口并保存双方的问答结果可以作为自己的问答词库的一个数据库累计.这些数据可以用于自己训练. 而最近希望获取一些语音资 ...

  8. 12. 集合类Collection和Map

    *:first-child { margin-top: 0 !important; } body>*:last-child { margin-bottom: 0 !important; } /* ...

  9. 【Java】You have an error in your SQL syntax ...

    详情如下: You have an error in your SQL syntax; check the manual that corresponds to your MySQL server v ...

  10. Apache NiFi 核心概念和关键特性

    本文来源于官方文档翻译 NiFi 的核心概念 NiFi 最早是美国国家安全局内部使用的工具,用来投递海量的传感器数据.后来由 apache 基金会开源.天生就具备强大的基因.NiFi基本设计理念与 F ...