`

IBATIS2知识点二:放出IBATIS2.3.4.8--守护IBATIS2

阅读更多

长话短说,偶在博客开出了IBATIS2栏目,为的是给守护IBATIS2的同僚以信心,勿盲目的跟风技术,蜻蜓点水(毕竟遗留项目升级不易)!

IBATIS2是有缺陷,但作为SQL MAP开发,拥有它足以!IBATIS3做了优化、做了调整,完全处于不兼容IBATIS2状态,但作为SQL MAP模式,若仅仅停留在代码重构、优化dynamic sql方面,也不值得我们去跟风!

我之所以力挺IBATIS这种SQL MAP模式,是应为它有“一站式的”ORM 框架所不及的应用环境(eg.以局外人的角度,不干预库表设计,完全透明化提供DAO层规范化服务),详见博文IBATIS2序:T[.WOLF]开场白

对于IBATIS的缺陷,大家谈的比较多的可能有以下几点:

①对数据库移植性支持不好;

②要自行编写sql,影响开发速度;

③数据库连接池方面做的不够;

④没有解决关联1:M/M:N存在N+1选择问题;

⑤分页采用逻辑分页,不支持大数据量选取,进而导致分页相关功能失效;

......

从相对论的角度来看,其实这些问题都不是什么大问题:

①对数据库移植性支持不好。有多少项目必须要支持多数据库呢?尤其是后台型项目!

②要自行编写sql,影响开发速度。“君子性非异也,善假于物也。”为何不借助于工具或自行写个代码生成器呢?自行写sql,其实这也正是SQL MAP灵活性的核心所在!

③数据库连接池方面做的不够。关于这个问题,我们完全可以借助于Spring来弥补啊,充分发挥Spring粘合剂的优势!

④没有解决关联1:M/M:N存在N+1选择问题。这个问题在IBATIS2.2.0借助于groupBy关键字都已经解决了,之所以会有朋友认为尚未解决我猜是受中文文档更新不及时误导所致,官方提供的中文文档iBATIS-SqlMaps-2_cn.pdf更新时间是2004年6月17日,而问题解决的英文文档iBATIS-SqlMaps-2_en.pdf更新时间是2006年8月9日!(注:并非利用解决N+1问题总是最好的,还要综合考虑,也许懒加载更适合)

⑤分页采用逻辑分页,不支持大数据量选取,进而导致分页相关功能失效。这的确是一个硬伤,但并不是无药可救。网上有很多朋友通过继承SqlExecutor实现物理分页,并通过反射替换应用中真实sql执行器(详见http://pengfeng.iteye.com/blog/200772http://blog.csdn.net/cyoubunketu/archive/2009/06/06/4246676.aspx--刚去google的)。

 

看到很多朋友为了⑤忙碌,要做IBATIS2的守护者,今天放出我对⑤的解决方案,和朋友们的做法大同小异,下面简单的介绍下。

发布版本:iBatis2.3.4.8;《见附件》

基础版本:iBatis2终结版(V2.3.4.726)

解决的问题:①支持数据库物理分页选取;②消除接口PaginatedList及实现类PaginatedDataList过时注释;
注意事项:①若您采用的是Oracle数据库,则对您完全透明,无需做任何调整;
                ②若您采用的是MySQL数据库,则需要修改dialect.properties属性文件;
                   文件的位置:com/ibatis/ext/sqlmap/engine/dialect/dialect.properties
                   修改的方法:将dialectClass=com.ibatis.ext.sqlmap.engine.dialect.Dialect4Oracle
                         改为dialectClass=com.ibatis.ext.sqlmap.engine.dialect.Dialect4MySQL
               ③若您采用其他数据库,则需要实现接口com.ibatis.ext.sqlmap.engine.dialect.Dialect,并按②修改dialect.properties属性文件。
附Oracle版的Dialect接口实现代码:

package com.ibatis.ext.sqlmap.engine.dialect;

public class Dialect4Oracle implements Dialect {

 public boolean supportPhysicLimit() {
  return true;
 }
 public String sqlLimitWrapper(String sql, long offset, long maxsize) {
  if (sql.indexOf("select") == -1)
   throw new RuntimeException(
     "the parameter sql of Dialect4Oracle.sqlWrapper is not a valid select sql:\n"
       + sql);
  sql = sql.trim();
  /* get fields of raw sql,if your ibatis version is lower than version2.2.0 
               or the ibatis version does not support more columns selected than mapped in conf.xml
  String sqlFields = sql
    .substring("select".length(), sql.indexOf("from"));
  */
  // construct rule sql
  StringBuffer dSQL = new StringBuffer(sql.length() + 100);
  //dSQL.append("select").append(sqlFields).append("from (");
  dSQL.append("select").append(" * ").append("from (");
  dSQL.append("select tab_.*,rownum rownum_ from (").append(sql).append(
    ")");
  dSQL.append(" tab_ where rownum<=").append(offset + maxsize);
  dSQL.append(")");
  dSQL.append(" where rownum_>").append(offset);
  //System.out.println(dSQL.toString());
  return dSQL.toString();
 }
}

 
附分页调用代码片段(非WEB模式):

PaginatedDataList pagerList = (PaginatedDataList) service.userDaoIntf.queryForPaginatedList("selectUserList", aUser, 10);
do {
 if(!pagerList.isEmpty()) {
  Iterator itr = pagerList.iterator();
  while(itr.hasNext()) {
   //do something,for example as follows
   System.out.println(((User)itr.next()).getUsername());
  }
 }
} while(pagerList.nextPage());

 
在IBATIS3尚未走向成熟,尚未得到多方支持,尚未表现出优秀的特征之前,请先沉住气,让我们共同守护IBATIS2,这也是我前不久开出IBATIS2栏目打算介绍IBATIS2的基础知识、最佳实践的初衷!

 

特别提示:以Hibernate为代表的一站式ORM思想不同于以IBTIS这种SQL Map方式的半自动化的ORM,两者各有千秋,选谁&用谁视实际项目情况而定,这是我一贯坚持认为的。关于这两个框架的相关知识都将会有所介绍,以便共享知识,相互交流。

分享到:
评论
14 楼 redsnow_fenglin 2011-08-15  
Speak-shuai 写道
bushkarl 写道
rar 解压缩密码多少呢?

我也想问这个问题

rar里面有注释哦:
解压码:http://redsnow-fenglin.javaeye.com/
13 楼 Speak-shuai 2011-05-19  
bushkarl 写道
rar 解压缩密码多少呢?

我也想问这个问题
12 楼 bushkarl 2011-02-18  
rar 解压缩密码多少呢?
11 楼 shiwei2006 2011-02-18  
支持楼主,自己可以根据实际项目情况再进行适当修改就行了。关于项目的兼容性和随时可以更换数据库这样的需求应该是很少的。
10 楼 redsnow_fenglin 2010-08-16  
joyfun 写道
看楼主的说明 主要加上了分页 其实可以以插件的形式发布 这样兼容性也还好
mybatis2.3.5已经发布了
http://code.google.com/p/mybatis/downloads/list?can=2


对于仍有第三方升级可能的情况,若我们添加扩展功能,则您的建议很好的!
对于已经是终版的第三方,若我们处理的是她的硬伤,则兼顾性能,对其源码升级应该是较好的选择。同时,这样以来可以顺便消除讨厌的分页过期注释!

此外,mybatis2.3.5(是ibatis2.3.4的升版,所谓的ibatis3,被直接更名了),关于她的分页是采用的您的思想--扩展插件,在javaeye上有两篇样例博文http://melin.iteye.com/blog/491713和http://badqiu.iteye.com/blog/493884!
9 楼 joyfun 2010-08-13  
看楼主的说明 主要加上了分页 其实可以以插件的形式发布 这样兼容性也还好
mybatis2.3.5已经发布了
http://code.google.com/p/mybatis/downloads/list?can=2
8 楼 songfantasy 2010-08-13  
支持ibatis2,3已经改名字了。晕
7 楼 redsnow_fenglin 2010-08-11  
<p>修正一个BUG: <br>因之前发布的ibatis2.3.4.8是在Application下测试通过的,但今天发现在WEB模式下由spring加载报错说加载不到dialectClass,原因是类加载器的问题,现已修正。</p>
<p>代码在com.ibatis.sqlmap.engine.impl.SqlMapExecutorDelegate的无参构造子中,修正前后分别为: <br>修正前:</p>
<pre name="code" class="java">                /**
* Default constructor
*/
public SqlMapExecutorDelegate() {
mappedStatements = new HashMap();
cacheModels = new HashMap();
resultMaps = new HashMap();
parameterMaps = new HashMap();

sqlExecutor = new EnhancedSqlExecutor();
InputStream is = ClassLoader
.getSystemClassLoader()
.getResourceAsStream(
"com/ibatis/ext/sqlmap/engine/dialect/dialect.properties");
Properties props = new Properties();
try {
props.load(is);
String dialectClass = props.getProperty("dialectClass");
((EnhancedSqlExecutor) sqlExecutor).setDialect((Dialect) Class
.forName(dialectClass).newInstance());

} catch (Exception e) {
throw new RuntimeException(
new StringBuilder(
"SqlMapExecutorDelegate not find the dialectClass in com/ibatis/ext/sqlmap/engine/dialect/dialect.properties,detail info:\n"
+ e.getMessage()).toString());
}
typeHandlerFactory = new TypeHandlerFactory();
dataExchangeFactory = new DataExchangeFactory(typeHandlerFactory);
}</pre>
<p> <br>修正后:</p>
<pre name="code" class="java">/**
* Default constructor
*/
public SqlMapExecutorDelegate() {
mappedStatements = new HashMap();
cacheModels = new HashMap();
resultMaps = new HashMap();
parameterMaps = new HashMap();

sqlExecutor = new EnhancedSqlExecutor();
String dialectINI = "com/ibatis/ext/sqlmap/engine/dialect/dialect.properties";
InputStream is = SqlMapExecutorDelegate.class.getClassLoader().getResourceAsStream(dialectINI);
if(is == null) {
is = Thread.currentThread().getContextClassLoader().getResourceAsStream(dialectINI);
if(is == null) {
is = ClassLoader.getSystemClassLoader().getResourceAsStream(dialectINI);
}
}
Properties props = new Properties();
try {
props.load(is);
String dialectClass = props.getProperty("dialectClass");
((EnhancedSqlExecutor) sqlExecutor).setDialect((Dialect) Class
.forName(dialectClass).newInstance());

} catch (Exception e) {
throw new RuntimeException(
new StringBuilder(
"SqlMapExecutorDelegate not find the dialectClass in com/ibatis/ext/sqlmap/engine/dialect/dialect.properties,detail info:\n"
+ e.getMessage()).toString());
}
typeHandlerFactory = new TypeHandlerFactory();
dataExchangeFactory = new DataExchangeFactory(typeHandlerFactory);
}</pre>
<p> <br>重新发布一份见附件!</p>
<p> </p>
6 楼 shihujie861028 2010-08-03  
使用Ibatis有段时间了,ibatis 还是挺不错的,支持的要!
5 楼 mqlfly2008 2010-08-03  
使用 ibatis ,支持
4 楼 鱼言风语 2010-08-01  
握个爪,找到同志了!

俺已经修改了几处CoolSQL,使其用起来更爽

现在在做一个修改,让其能支持多表查询自动生成BO、DAO、SqlMap
3 楼 aimer311 2010-08-01  
上个月我还在很爽的使用abator生成代码,现在正徘徊在hibernate的OneToMany,ManyToMany中。我们掌控的越少,不可预料的结果就越会发生。
还是比较习惯sql的感觉。
2 楼 Brera 2010-07-31  

手上有中英文的文档,一直没有去读英文的
刚刚去看了才发现,确实不同
今晚得好好读读
1 楼 lyb520320 2010-07-31  
好帖,我是IBatis的忠实拥护者,共同进步!

相关推荐

Global site tag (gtag.js) - Google Analytics