不使用DalFactory和IDAL,支持多种数据库应用
MS的PetShop示例应用程序的“多层架构”被很多.NET开发人员奉为经典的架构,我以前做的项目团队的Leader也是照搬它的,甚至来到现在这个公司后,好几个新来的同事建解决方案也是照搬PetShop的架构,可见PetShop对大家影响之深。
下面是PetShop 3.0的架构图:
为了支持多数据库应用,在DAL中,定义了DAL Interface和DAL Factory,下面这个图也许跟简单直观一些(引用Do you know, jack? 园友http://www.cnblogs.com/zq8024/archive/2011/03/31/2001436.html 中的图片,请见谅:))
通过这个图大家都能够明白,引入DalFactory和IDAL就是为了系统支持不同的数据库。
PDF.NET数据开发框架采用了完全不同的方式,先看看它的分层架构图:
如果在DAL中没有某种数据库特有的SQL语句,DalFactory是不需要的,当然IDAL也不需要了。例如ORM操作,一般不会用到数据库的特性,发出的都是标准的SQL语句。PDF.NET数据开发框架的ORM操作是通过EntityQuery和OQL表达式来实现,在具体支持不同数据库的时候,底层采用的是反射工厂模式:
/// <summary>
/// 创建公共数据访问类的实例
/// </summary>
/// <param name="providerAssembly">提供这程序集名称</param>
/// <param name="providerType">提供者类型</param>
/// <returns></returns>
public static AdoHelper CreateInstance( string providerAssembly, string providerType )
{
Assembly assembly = Assembly.Load( providerAssembly );
object provider = assembly.CreateInstance( providerType );
if( provider is AdoHelper )
{
return provider as AdoHelper;
}
else
{
throw new InvalidOperationException("当前指定的的提供程序不是 AdoHelper 抽象类的具体实现类,请确保应用程序进行了正确的配置(如connectionStrings 配置节的 providerName 属性)。");
}
}
这样只需要在配置文件中进行配置,指明采用何种数据库即可,这是框架脱离DalFactory+IDAL的第一种方式。
当然,为了高效的使用某种数据库的特性,有可能会写一些数据库特性的SQL,要使得系统支持不同的数据库,还得使用DalFactory,因此得定义IDAL。
PDF.NET数据开发框架为了解决这个问题,将所有的SQL语句写在一个配置文件SqlMap.config中,使用工具自动生成框架的DAL代码,即SqlMapDAL,不同的数据库系统使用不同的SqlMap.config文件即可,不需要替换SqlMapDAL,因此,框架再也不需要定义DalFactory和IDAL了,这应该算是第二种方式。 下图是根据SqlMap自动生成代码并运行的流程:
在SqlMap中,可以将结果映射成DataSet,实体类和实体类集合,也可以是单值类型,可以完成各种复杂的SQL操作,可以处理存储过程。系统将SQL语句中的参数映射成DAL代码中方法的参数,使得操作非常直观高效,并且没有SQL注入的问题。
PDF.NET数据开发框架通过自己的ORM(EntityQuery+OQL)结合SQL-MAP的方式,使得喜欢OO的人和喜欢SQL的人都能找到自己需要的,便利性和灵活性都能够兼得。
详细请看官网介绍:http://www.pwmis.com/sqlmap
或者看我博客中的相关文章介绍。
- PyQt5 GUI应用程序工具包入门(2)
- 用Python研究了三千套房子,告诉你究竟是什么抬高了房价?
- 设计模式之装饰模式
- Android网络编程(六)OkHttp3用法全解析
- Android网络编程(五)OkHttp用法全解析
- Android网络编程(一)HTTP协议原理
- Spring Cloud实战小贴士:Zuul统一异常处理(二)
- Android View体系(九)自定义View
- Android网络编程(二)HttpClient与HttpURLConnection
- Spring Batch入门篇
- Android网络编程(三)Volley用法全解析
- Android网络编程(八)源码解析OkHttp中篇[复用连接池]
- Mybatis SqlSessionTemplate 源码解析
- MapperScannerConfigurer处理过程源码分析
- JavaScript 教程
- JavaScript 编辑工具
- JavaScript 与HTML
- JavaScript 与Java
- JavaScript 数据结构
- JavaScript 基本数据类型
- JavaScript 特殊数据类型
- JavaScript 运算符
- JavaScript typeof 运算符
- JavaScript 表达式
- JavaScript 类型转换
- JavaScript 基本语法
- JavaScript 注释
- Javascript 基本处理流程
- Javascript 选择结构
- Javascript if 语句
- Javascript if 语句的嵌套
- Javascript switch 语句
- Javascript 循环结构
- Javascript 循环结构实例
- Javascript 跳转语句
- Javascript 控制语句总结
- Javascript 函数介绍
- Javascript 函数的定义
- Javascript 函数调用
- Javascript 几种特殊的函数
- JavaScript 内置函数简介
- Javascript eval() 函数
- Javascript isFinite() 函数
- Javascript isNaN() 函数
- parseInt() 与 parseFloat()
- escape() 与 unescape()
- Javascript 字符串介绍
- Javascript length属性
- javascript 字符串函数
- Javascript 日期对象简介
- Javascript 日期对象用途
- Date 对象属性和方法
- Javascript 数组是什么
- Javascript 创建数组
- Javascript 数组赋值与取值
- Javascript 数组属性和方法
- MySQL 切换数据库、用户卡死:“You can turn off this feature to get a quicker startup with -A“处理方法
- MySQL 数据库mysqlbinlog使用问题:unknown variable ‘default-character-set=utf8‘.解决方法
- Python 技术篇-pip安装提示:‘pip‘ 不是内部或外部命令,也不是可运行的程序或批处理文件,问题解决方法
- Jaskson精讲第6篇-自定义JsonSerialize与Deserialize实现数据类型转换
- let和var和const
- Jupyter 编写python代码实现代码自动补齐功能设置实例演示
- 第37期:从头学二叉搜索树(面试常考)
- Jupyter 工具的安装与使用方法,jupyter运行python代码演示,好用的python编辑器推荐!
- Nginx相关配置与操作
- Python 技术篇-全局与当前socket超时连接时间设置方法实例演示,查看socket超时连接时间
- 给 JDK 报了一个 P4 的 Bug,结果居然……
- Python 套接字-判断socket服务端有没有关闭的方法实例演示,查看socket运行状态
- docker安装logstash
- Rook Operator 源码分析(1) - osd 启动的流程
- Python 技术篇-利用pyqt5库监听剪切板变动,clipboard.dataChanged.connect()剪切板监听