名字起得有点夸张了,其实就是实现 基于 Data Access Application Block的DAL基类和约束 首先Repository部分没什么好描述的,如果有不了解的可以直接百度或者谷歌相关内容,直接上具体代码 注意此部分没有写批量查询的方法(比如FindAll,这跟后面的基类设定
名字起得有点夸张了,其实就是实现基于Data Access Application Block的DAL基类和约束
首先Repository部分没什么好描述的,如果有不了解的可以直接百度或者谷歌相关内容,直接上具体代码
注意此部分没有写批量查询的方法(比如FindAll,这跟后面的基类设定有关)
/// <summary> /// DataAccess Repository /// </summary> /// <typeparam name="T1"></typeparam> /// <typeparam name="T2"></typeparam> public interface IRepository<t1 t2> { /// <summary> /// 根据主键获取对应的实体对象 /// </summary> /// <param name="key"> /// <returns></returns> T1 GetEntityByKey(T2 key); /// <summary> /// 单个新增 /// </summary> /// <param name="entity"> /// <returns></returns> bool Insert(T1 entity); /// <summary> /// 单个编辑 /// </summary> /// <param name="entity"> /// <returns></returns> bool Update(T1 entity); /// <summary> /// 单个删除 /// </summary> /// <param name="key"> /// <returns></returns> bool Delete(T2 key); }</t1>然后是DAL抽象基类,该基类设计上实现读写分离,而且每个子类只应有一个数据库连接,这样做的隐藏目的是每个DAL类都只应该执行自己最基本的功能:只关心数据库交互,不关心业务逻辑
/// <summary> /// DAL基类,<strong>基于</strong>EntLib,读写分离 /// </summary> public abstract class DataAccessBase { private Database _readDB; private Database _writeDB; /// <summary> /// 要使用的读数据库配置节,如果配置为null或者空则会调用默认配置节 /// </summary> protected abstract string ReadDBName { get; } /// <summary> /// 要使用的写数据库配置节,如果配置为null或者空则会调用默认配置节 /// </summary> protected abstract string WriteDBName { get; } /// <summary> /// 读库 /// </summary> protected Database ReadDB { get { if (this._readDB == null) { this._readDB = this.GetDatabase(this._writeDB, this.ReadDBName); } return this._readDB; } } private Database GetDatabase(Database db, string dbName) { if (this.ReadDBName == this.WriteDBName && db != null) { return db; } else { return this.CreateDatabase(dbName); } } /// <summary> /// 写库 /// </summary> protected Database WriteDB { get { if (this._writeDB == null) { this._writeDB = this.GetDatabase(this._readDB, this.WriteDBName); } return this._writeDB; } } private Database CreateDatabase(string dbName) { DatabaseProviderFactory factory = new DatabaseProviderFactory(); if (string.IsNullOrWhiteSpace(dbName)) { return factory.CreateDefault(); } else { return factory.Create(dbName); } } }最后就是IRepository接口与DataAccessBase的组合实现SingleDataAccessBase,为什么不在DataAccessBase时就实现IRepository呢?因为设计上DataAccessBase是可以同时运用于多表操作及单表操作的,多表操作时,IRepository不存在任何意义,只有单表操作时,IRepository才有意义,而SingleDataAccessBase就是单表DAL基类
using System.Collections.Generic; using System.Linq; using Microsoft.Practices.EnterpriseLibrary.Data; using System.Data.Common; /// <summary> /// 单表DataAccess基类,所有单表DataAccess应当继承此类,建议非共用部分同样实现接口 /// 多表但单数据库操作的DataAccess不能继承此类,而应继承DataAccessBase /// </summary> /// <typeparam name="T1"></typeparam> /// <typeparam name="T2"></typeparam> public abstract class SingleDataAccessBase<t1 t2> : DataAccessBase, IRepository<t1 t2> where T1 : new() { #region SqlString /// <summary> /// GetEntityByKey方法对应的Sql语句 /// </summary> protected abstract string SelectSql { get; } /// <summary> /// Insert方法对应的Sql语句 /// </summary> protected abstract string InsertSql { get; } /// <summary> /// Update方法对应的Sql语句 /// </summary> protected abstract string UpdateSql { get; } /// <summary> /// Delete方法对应的Sql语句 /// </summary> protected abstract string DeleteSql { get; } #endregion #region IRepository<t1> 成员 /// <summary> /// 根据主键获取对应的实体对象 /// </summary> /// <param name="key"> /// <returns></returns> public virtual T1 GetEntityByKey(T2 key) { return this.ReadDB.ExecuteBySqlString(this.SelectSql, null, (IRowMapper<t1>)null, this.GetKeyParameters(key).ToArray()).FirstOrDefault(); } /// <summary> /// 单个新增,如果是自增主键,则需要override /// </summary> /// <param name="entity"> /// <returns></returns> public virtual bool Insert(T1 entity) { return this.WriteDB.ExecuteNonQueryBySqlString(this.InsertSql, (cmd) => { this.SetParametersByEntity(entity, cmd); }) > 0; } /// <summary> /// 单个编辑 /// </summary> /// <param name="entity"> /// <returns></returns> public virtual bool Update(T1 entity) { return this.WriteDB.ExecuteNonQueryBySqlString(this.UpdateSql, (cmd) => { this.SetParametersByEntity(entity, cmd); }) > 0; } /// <summary> /// 单个删除 /// </summary> /// <param name="key"> /// <returns></returns> public virtual bool Delete(T2 key) { return this.WriteDB.ExecuteNonQueryBySqlString(this.DeleteSql, (cmd) => { this.SetParametersByKey(key, cmd); }) > 0; } #endregion #region Methods /// <summary> /// 通过Entity填充DbParameter /// </summary> /// <param name="entity"> /// <param name="cmd"> protected abstract void SetParametersByEntity(T1 entity, DbCommand cmd); /// <summary> /// 通过Key填充DbParameter /// </summary> /// <param name="key"> /// <param name="cmd"> protected virtual void SetParametersByKey(T2 key, DbCommand cmd) { var paramters = this.GetKeyParameters(key); cmd.Parameters.AddRange(paramters.ToArray()); } /// <summary> /// 通过协变的方式根据Key获取对应的DbParameter /// </summary> /// <param name="key"> /// <returns></returns> protected abstract IEnumerable<dbparameter> GetKeyParameters(T2 key); #endregion }</dbparameter></t1></t1></t1></t1>
这里SingleDataAccessBase其实并不能算实现了IRepository,只是进行了相关的约束规范而已,实际子类需要提供相应的SqlString以及Parameter实现(注意此处用到了前面博客中写的微软企业库扩展,具体见[EntLib]微软企业库6 Data Access Application Block 扩展),因为Data Access Application Block其实并不是一个ORM,如果采用这个类库,其实已经有比较大的概率可以确定系统对性能的要求较高,当然你也可以通过反射之类的实现真正的Repository(前提是你的POCO与关系型数据库中字段能对应上)
SingleDataAccessBase子类建议划分区域块,对应的代码写入对应区域,如
#region ConnectionSetting protected override string ReadDBName { get { throw new NotImplementedException(); } } protected override string WriteDBName { get { throw new NotImplementedException(); } } #endregion #region SqlString protected override string SelectSql { get { throw new NotImplementedException(); } } protected override string InsertSql { get { throw new NotImplementedException(); } } protected override string UpdateSql { get { throw new NotImplementedException(); } } protected override string DeleteSql { get { throw new NotImplementedException(); } } #endregion #region Read #endregion #region Write #endregion #region DbParameter protected override void SetParametersByEntity(T1 entity, DbCommand cmd) { throw new NotImplementedException(); } protected override IEnumerable<dbparameter> GetKeyParameters(T2 key) { throw new NotImplementedException(); } #endregion</dbparameter>
可能很多人觉得区分多表操作的DAL和单表操作的DAL没有必要或者没有意义,的确,在功能上,写在一个里面和写在多个里面没有区别,系统较小时还没有什么问题,如果系统较大,经由的开发人员较多时,很可能会出现某个开发人员想要用到某个表的某些字段,但因为系统较大又没有相关约束,导致该开发人员不知道去哪里找是否该方法已经存在,为了简易起见,最常见的做法是直接在自己需要的地方新写这部分代码,而最终的后果就是,代码分布越来越混乱,当表结构发生变化时,此部分尤其变成了灾难,因为如果不通过查表名,你根本无法预见到底哪些地方用到了这张表!而区分单表及多表操作,就是要解决这样的问题,当表结构发生变化时,也只需要找该表对应的单表DAL以及以数据库对应的多表DAL两个地方
最后吐槽下,企业库5真心伤不起,为了用DataAccess模块,连连千千添加了5个相关dll引用,真心怪不得人家不愿意用,还好6没了这个问题……

微软宣布进一步扩展和 Meta 的 AI 合作伙伴关系,Meta 已选择 Azure 作为战略性云供应商,以帮助加速 AI 研发。在 2017 年,微软和 Meta(彼时还被称为 Facebook)共同发起了 ONNX(即 Open Neural Network Exchange),一个开放的深度学习开发工具生态系统,旨在让开发者能够在不同的 AI 框架之间移动深度学习模型。2018 年,微软宣布开源了 ONNX Runtime —— ONNX 格式模型的推理引擎。作为此次深化合作的一部分,Me

OTO 是业内首个自动化、一站式、用户友好且通用的神经网络训练与结构压缩框架。 在人工智能时代,如何部署和维护神经网络是产品化的关键问题考虑到节省运算成本,同时尽可能小地损失模型性能,压缩神经网络成为了 DNN 产品化的关键之一。DNN 压缩通常来说有三种方式,剪枝,知识蒸馏和量化。剪枝旨在识别并去除冗余结构,给 DNN 瘦身的同时尽可能地保持模型性能,是最为通用且有效的压缩方法。三种方法通常来讲可以相辅相成,共同作用来达到最佳的压缩效果。然而现存的剪枝方法大都只针对特定模型,特定任务,且需要很

ChatGPT在手,有问必答。你可知,与它每次对话的计算成本简直让人泪目。此前,分析师称ChatGPT回复一次,需要2美分。要知道,人工智能聊天机器人所需的算力背后烧的可是GPU。这恰恰让像英伟达这样的芯片公司豪赚了一把。2月23日,英伟达股价飙升,使其市值增加了700多亿美元,总市值超5800亿美元,大约是英特尔的5倍。在英伟达之外,AMD可以称得上是图形处理器行业的第二大厂商,市场份额约为20%。而英特尔持有不到1%的市场份额。ChatGPT在跑,英伟达在赚随着ChatGPT解锁潜在的应用案

随着OpenAI DALL-E和Midjourney的推出,AI艺术生成器开始变得越来越流行,它们接受文本提示并将其变成美丽的、通常是超现实的艺术品——如今,有两家大企业加入了这一行列。微软宣布,将通过Bing Image Creator把由DALL-E模型提供支持的AI图像生成功能引入Bing搜索引擎和Edge浏览器。创意软件开发商Adobe也透露,将通过名为Firefly的AI艺术生成产品来增强自己的工具。对于有权访问Bing聊天预览的用户来说,这一新的AI图像生成器已经可以在“创意”模式下

自然语言处理(NLP)模型读不懂人话、将文本理解为相反的意思,是业界顽疾了。 现在微软表示,开发出解决此弊的方法。微软开发AdaTest方法来测试NLP模型 可作为跨越各种应用基础的大型模型,或称平台模型的进展已经大大改善了AI处理自然语言的能力。但自然语言处理(NLP)模型仍然远不完美,有时会以令人尴尬的方式暴露缺陷。 例如有个顶级的商用模型,将葡萄牙语中的「我不推荐这道菜」翻译成英语中的「我非常推荐这道菜」。 这些失败之所以继续存在,部分原因是寻找和修复NLP模型中的错误很难,以至于严重的

大家好,我是菜鸟哥!最近逛G网,发现微软开源了一个项目叫「playwright-python」,作为一个兴起项目。Playwright 是针对 Python 语言的纯自动化工具,它可以通过单个API自动执行 Chromium,Firefox 和 WebKit 浏览器,连代码都不用写,就能实现自动化功能。虽然测试工具 selenium 具有完备的文档,但是其学习成本让一众小白们望而却步,对比之下 playwright-python 简直是小白们的神器。Playwright真的适用于Python吗?

微软必应完善文字生成图像能力,Adobe 今日也发布 Firefly,杀入生成式 AI 这场游戏。 昨晚实在是有些热闹。一边英伟达 GTC 正在进行中,一边谷歌正式开放了 Bard 的测试,这里微软必应也不甘寂寞。今日,微软正式宣布,必应搜索引擎接入了 OpenAI 的 DALL·E 模型,增加了 AI 生成图像的功能。也就是说,在接入 ChatGPT 之后,必应再次强化,Bing Image Creator 能够让用户用 DALL·E 模型生成图像。「对于拥有必应预览版权限的用户,Bing I

近日微软推出了Security Copilot,这款新工具旨在通过AI助手简化网络安全人员的工作,帮助他们应对安全威胁。 网络安全人员往往要管理很多工具,和来自多个来源的海量数据。近日微软宣布推出了Security Copilot,这款新工具旨在通过AI助手简化网络安全人员的工作,帮助他们应对安全威胁。Copilot利用基于OpenAI的GPT-4最新技术,让网络安全人员能够就当前影响环境的安全问题提问并获得答案,甚至可以直接整合公司内部的知识,为团队提供有用的信息,从现有信息中进行学习,将当前


Hot AI Tools

Undresser.AI Undress
AI-powered app for creating realistic nude photos

AI Clothes Remover
Online AI tool for removing clothes from photos.

Undress AI Tool
Undress images for free

Clothoff.io
AI clothes remover

AI Hentai Generator
Generate AI Hentai for free.

Hot Article

Hot Tools

Safe Exam Browser
Safe Exam Browser is a secure browser environment for taking online exams securely. This software turns any computer into a secure workstation. It controls access to any utility and prevents students from using unauthorized resources.

PhpStorm Mac version
The latest (2018.2.1) professional PHP integrated development tool

ZendStudio 13.5.1 Mac
Powerful PHP integrated development environment

SublimeText3 Linux new version
SublimeText3 Linux latest version

Notepad++7.3.1
Easy-to-use and free code editor
