使用Castle扩展Ibatis.Net,面向接口编程-更优雅的代码
使用Ibatis.Net做项目半年了,甚是喜欢,感觉确实是个简单、轻巧的O/R Mapping框架,特别是将Sql配置在Xml文件中,相当于直接将Dao层抽离了出来。
本文假定读者对Ibatis.Net有一定的了解。
最近试用了一下Ibatis.Net的亲兄弟--Java的Mybatis,一对比发现:
执行一个查询,Ibatis.Net是这么写的:IList<UserEntity> list = SqlMapper.QueryForList<UserEntity>(prefix+ ".GetByFilter", parameters);
而Java的Mybatis是这么写的:List<UserEntity> list = dao.GetByFilter(parameters);
发现了没,后者的显然更优雅。
Mybatis之所以能这么调用,是因为Mybatis提供了一种面向接口编程的方法,只要写好接口,接口的方法名与map文件中sql片段的id相同,我们就能够直接通过接口调用。
我想了又想...C#也能够实现这样优雅的调用啊,可是为啥Ibatis.Net不提供呢,想到这,我开始明白Ibatis.Net是后妈生的。。。
说到这,进入主题吧,既然Ibatis.Net先天不够强大,那我们后天弥补吧,这里主要使用Castle这个组件来动态实现接口。
接下来我们做个Demo
1.搭建Ibatis.Net环境,这里就不说啦(最新版Ibatis.Net下载地址:http://download.csdn.net/detail/tzjzcy/7829759 )
2.引用Castle.Core.dll,这个dll实际上最新版的Ibatis.Net本身就有用到
3.创建一个测试表,录入数据,本文以mysql为例,代码如下:
CREATE TABLE `user`(
`Userid` INT NOT NULL AUTO_INCREMENT,
`Username` VARCHAR(100),
`Age` INT,
`City` VARCHAR(100), PRIMARY KEY (`Userid`)
); INSERT INTO `testex`.`user` (`Username`, `Age`, `City`) VALUES ('羊望', '', '厦门');
INSERT INTO `testex`.`user` (`Userid`, `Username`, `Age`, `City`) VALUES ('', '测试', '', '福州');
4.编写对应实体类
public class UserEntity
{
public int? Userid { get; set; } public string Username { get; set; } public int? Age { get; set; } public string City { get; set; }
}
5.写个简单的map文件:UserMap.xml
<?xml version="1.0" encoding="utf-8" ?>
<!--这里的namespace必须对应Dao接口的完整类名-->
<sqlMap namespace="IbatisExTest.Daos.IUserDao"
xmlns="http://ibatis.apache.org/mapping"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance">
<alias>
<typeAlias alias="UserEntity" type="IbatisExTest.Entities.UserEntity,IbatisExTest" />
</alias> <statements> <select id="GetByUserid" parameterClass="String" resultClass="UserEntity">
SELECT *
FROM user
<dynamic prepend="WHERE">
Userid =#value#
</dynamic>
</select> <select id="GetByFilter" parameterClass="Hashtable" resultClass="UserEntity">
SELECT *
From user
<dynamic prepend="WHERE">
<isNotEmpty prepend="AND" property="Userid">
Userid =#Userid#
</isNotEmpty>
<isNotEmpty prepend="AND" property="Username">
Username =#Username#
</isNotEmpty>
</dynamic>
</select> <insert id="InsertUser" parameterClass="UserEntity">
INSERT INTO user
( Username
, Age
, City)
VALUES (
#Username#
, #Age#
, #City#);
</insert> </statements>
</sqlMap>
6.写一个接口,接口的全名(命名空间+接口名)必须与map文件的namespace相同,接口的方法与map文件中的sql片段id对应
public interface IUserDao
{ UserEntity GetByUserid(string userid); IList<UserEntity> GetByFilter(Hashtable ht); object InsertUser(UserEntity user);
}
7.写一个BaseDao,作为动态创建的Dao实现类的基类,定义一个属性,传入SqlMapper用
public class BaseDao
{
public BaseDao(ISqlMapper sqlMapper)
{
this.SqlMapper = sqlMapper;
}
public ISqlMapper SqlMapper { get; private set; }
}
8.重点来了,编写Dao实现类的具体方法实现,通过Castle组件实现的,作用是:在调用接口的方法时,执行map中对应的sql片段
/// <summary>
/// Dao接口的方法实现
/// </summary>
public class DaoInterceptor : IInterceptor
{
public void Intercept(IInvocation invocation)
{
BaseDao baseDao = (BaseDao)invocation.Proxy;
//从基类BaseDao获取sqlMapper实例
ISqlMapper sqlMapper = baseDao.SqlMapper;
MethodInfo method = invocation.Method;
if (method.DeclaringType == null) return;
//获取接口的全名,即map文件的Namespace
string mapNamespace = method.DeclaringType.FullName;
//得到要执行的sql的完整id
string statementId = mapNamespace + "." + method.Name;
IMappedStatement ms = sqlMapper.GetMappedStatement(statementId);
if (ms is SelectMappedStatement)
{
ProcessSelectStatement(invocation, sqlMapper, statementId);
}
else if (ms is InsertMappedStatement)
{
ProcessInsertStatement(invocation, sqlMapper, statementId);
}
else if (ms is UpdateMappedStatement)
{
ProcessUpdateStatement(invocation, sqlMapper, statementId);
}
else if (ms is DeleteMappedStatement)
{
ProcessDeleteStatement(invocation, sqlMapper, statementId);
}
} private static void ProcessSelectStatement(IInvocation invocation, ISqlMapper sqlMapper, string statementId)
{
MethodInfo method = invocation.Method;
if (method.ReturnType.IsGenericType) //判断方法的返回值,如果是泛型,表示返回值是泛型集合
{
//通过反射调用sqlMapper.QueryForList方法
Type t = typeof(List<>).MakeGenericType(method.ReturnType.GetGenericArguments());
var list = Activator.CreateInstance(t);
MethodInfo miQueryForList = typeof(ISqlMapper).GetMethod("QueryForList",
new Type[] { typeof(string), typeof(object), typeof(List<>) });
miQueryForList.Invoke(sqlMapper, new object[] { statementId, invocation.Arguments[], list });
invocation.ReturnValue = list;
}
else //返回单个对象,或int等基本类型
{
//直接调用sqlMapper.QueryForObject方法
invocation.ReturnValue = sqlMapper.QueryForObject(statementId, invocation.Arguments[]);
}
} private static void ProcessInsertStatement(IInvocation invocation, ISqlMapper sqlMapper, string statementId)
{
invocation.ReturnValue = sqlMapper.Insert(statementId, invocation.Arguments[]);
} private static void ProcessUpdateStatement(IInvocation invocation, ISqlMapper sqlMapper, string statementId)
{
invocation.ReturnValue = sqlMapper.Update(statementId, invocation.Arguments[]);
} private static void ProcessDeleteStatement(IInvocation invocation, ISqlMapper sqlMapper, string statementId)
{
invocation.ReturnValue = sqlMapper.Delete(statementId, invocation.Arguments[]);
}
9.编写SqlMapper的扩展方法,动态实现Dao接口
public static class SqlMapExtensionMethods
{
/// <summary>
/// 获取Dao的实现
/// </summary>
/// <typeparam name="T">Dao接口</typeparam>
/// <param name="sqlMapper">sqlMapper</param>
/// <returns>返回Dao的实现</returns>
public static T GetDao<T>(this ISqlMapper sqlMapper)
{
ProxyGenerator generator = new ProxyGenerator();
DaoInterceptor daoInterceptor = new DaoInterceptor();
//创建一个BaseDao的代理类,并实现指定Dao接口
object proxy = generator.CreateClassProxy(typeof(BaseDao), new Type[] { typeof(T) }, ProxyGenerationOptions.Default, new object[] { sqlMapper }, daoInterceptor);
return (T)proxy;
}
}
10.这样就完成了扩展,让我们看看调用实例吧
class Program
{
private const string mapperNamespace = "IbatisExTest.Daos.IUserDao"; private static ISqlMapper SqlMapper
{
get { return Mapper.Get(); }
} private static IUserDao UserDao
{
get { return SqlMapper.GetDao<IUserDao>(); }
} static void Main()
{
Hashtable ht = new Hashtable();
ht["Username"] = "羊望";
//传统用法
IList<UserEntity> list1 = SqlMapper.QueryForList<UserEntity>(mapperNamespace + ".GetByFilter", ht); //新用法(代码更优雅了吧)
IList<UserEntity> list2 = UserDao.GetByFilter(ht); //测试新增
//UserEntity user = new UserEntity { Username = "新用户", Age = , City = "新城市" };
//UserDao.InsertUser(user);
}
}
最后,我们看到,扩展后我们只需要调用Dao接口的方法,代码更简洁了。
至于要比传统用法多写个Dao接口,这个工作或许我们可以通过代码生成工具来做吧。
源代码下载:http://files.cnblogs.com/lookup/Castle%E6%89%A9%E5%B1%95IbatisNet%E4%BE%8B%E5%AD%90.zip
欢迎拍砖:)
使用Castle扩展Ibatis.Net,面向接口编程-更优雅的代码的更多相关文章
- Castle扩展Ibatis.Net
使用Castle扩展Ibatis.Net,面向接口编程-更优雅的代码 使用Ibatis.Net做项目半年了,甚是喜欢,感觉确实是个简单.轻巧的O/R Mapping框架,特别是将Sql配置在Xml文件 ...
- 9条消除if...else的锦囊妙计,助你写出更优雅的代码
前言 最近在做代码重构,发现了很多代码的烂味道.其他的不多说,今天主要说说那些又臭又长的if...else要如何重构. 在介绍更更优雅的编程之前,让我们一起回顾一下,不好的if...else代码 一. ...
- 更优雅的清除浮动float方法
上篇文章是利用 :after 方法清除浮动float(作用于浮动元素的父元素上). ; } //为了兼容性,因为ie6/7不能使用伪类,所以加上此行代码. .outer:after {;;visibi ...
- 设计与实现分离——面向接口编程(OO博客第三弹)
如果说继承是面向对象程序设计中承前启后的特质,那么接口就是海纳百川的体现了.它们都是对数据和行为的抽象,都是对性质和关系的概括.只不过前者是纵向角度,而后者是横向角度罢了.今天呢,我想从设计+语法角度 ...
- C#扫盲篇(二)依赖倒置•控制反转•依赖注入•面向接口编程--满腹经纶的说
扫盲系列的文章收到了广大粉丝朋友的支持,十分感谢,你们的支持就是我最大动力. 我的扫盲系列还会继续输出,本人也是一线码农,有什么问题大家可以一起讨论.也可以私信或者留言您想要了解的知识点,我们一起进步 ...
- C#进阶系列——MEF实现设计上的“松耦合”(终结篇:面向接口编程)
序:忙碌多事的八月带着些许的倦意早已步入尾声,金秋九月承载着抗战胜利70周年的喜庆扑面而来.没来得及任何准备,似乎也不需要任何准备,因为生活不需要太多将来时.每天忙着上班.加班.白加班,忘了去愤,忘了 ...
- C#面向接口编程详解(1)——思想基础
我想,对于各位使用面向对象编程语言的程序员来说,“接口”这个名词一定不陌生,但是不知各位有没有这样的疑惑:接口有什么用途?它和抽象类有什么区别?能不能用抽象类代替接口呢?而且,作为程序员,一定经常听到 ...
- 面向接口编程详解-Java篇
相信看到这篇文字的人已经不需要了解什么是接口了,我就不再过多的做介绍了,直接步入正题,接口测试如何编写.那么在这一篇里,我们用一个例子,让各位对这个重要的编程思想有个直观的印象.为充分考虑到初学者,所 ...
- IOC解耦-面向接口编程的优点
原文:https://blog.csdn.net/jj_nan/article/details/70161086 参考:https://www.cnblogs.com/landeanfen/p/477 ...
随机推荐
- 启示—地点IT高管20在职场心脏经(读书笔记6)
启示--一个IT高管20在职场心脏经 第七章 关于销售 用"最"来形容公司的销售.能够用上若干的词汇: 最牛,最累,最精,最傻,最有钱,最贱,最能吹.最能装... 1.1 销售 ...
- 在ASP.NET 5应用程序中的跨域请求功能详解
在ASP.NET 5应用程序中的跨域请求功能详解 浏览器安全阻止了一个网页中向另外一个域提交请求,这个限制叫做同域策咯(same-origin policy),这组织了一个恶意网站从另外一个网站读取敏 ...
- 网站的SEO以及它和站长工具的之间秘密(转)
博客迁移没有注意 URL 地址的变化,导致百度和 google 这两只爬虫引擎短时间内找不到路.近段时间研究了下国内最大搜索引擎百度和国际最大搜索引擎google的站长工具,说下感受. 百度的站长工具 ...
- 学习日记之单例模式和Effective C++
单例模式(Singleton):保证一个类仅有一个实例,并提供一个訪问它的全局訪问点. (1),通常我们能够让一个全局变量使得一个对象被訪问,但它不能防止你实例化多个对象.一个最好的办法就是,让类自身 ...
- 霍夫曼(最优二叉树)和Java达到
一.定义 一些定义: 节点之间的路径长度:在从节点树中的一个节点也经历分公司,这构成的两个节点之间的路径分支的数目后这就是所谓的路径长度 的路径长度:从树的根节点到树中每一结点的路径长度之和. 在结点 ...
- 提高你的Java代码质量吧:推荐在复杂字符串操作中使用正则表达式
一.分析 字符串的操作,诸如追加.合并.替换.倒序.分隔等,都是在编码过程中经常用到的,而且Java也提供了append.replace.reverse.split等方法来完成这些操作,它们使用起来 ...
- SQL Server AlwaysON 同步模式的疑似陷阱
原文:SQL Server AlwaysON 同步模式的疑似陷阱 SQL Server 2012 推出的最重要的功能之一Alwayson,是一个集之前Cluster和Mirror于一体的新功能,即解决 ...
- 【百度地图API】如何利用地图API制作汽车沿道路行驶的动画?——如何获得道路层数据
原文:[百度地图API]如何利用地图API制作汽车沿道路行驶的动画?--如何获得道路层数据 有几个做汽车导航的朋友问我说,他们想在地图上制作一辆车沿着道路行驶的动画.可是,百度地图的道路数据并没有公开 ...
- Swift语言指南(四)--类型安全和类型推断
原文:Swift语言指南(四)--类型安全和类型推断 Swift是一门类型安全语言,类型安全语言需要代码里值的类型非常明确.如果你的代码中有部分值需要String类型,你就不能错误地传递Int. 鉴于 ...
- .NET Framework 各版本区别
.NET Framework 各版本区别 .NET Framework 1.1 自1.0版本以来的改进:自带了对mobile asp .net控件的支持.这在1.0版本是以附加功能方式实现的,现在已经 ...