JR 精品文章 - 让spring帮助你在MVC层解决JPA的缓迟加载问题
AD: jr (at) javaresearch.org


首页 | 动态 | 文章 | FAQ  | 新闻 | 下载 | 代码 | 工作 | 调查 | 术语 | 站点 | 图书 | 论坛 | 帮助 | 全部  

TOP | 交流 | 软件 | 专栏 | 开源 | 译/著 | 源码 | API  | 推荐 | FTP  | 积分 | 统计 | 搜索 | Blog | 我们  
首页 » 研究文集 » 开发框架 搜索标题相关文章 搜索标题相关文章    评论此文章 发表评论     开始监控此文章 开始监控   加入收藏夹  加入收藏夹
让spring帮助你在MVC层解决JPA的缓迟加载问题
easyjf 原创   更新:2007-03-28 02:16:13  版本: 1.0   

      作为EJB3.0的一部分,JPA是一个好东西。其简单的配置方式及强大的默认配置支持,使其可以轻松自由的存在于轻量与重量之间,如果现在您的JavaEE项目,不管是选择轻量级构架还是重量级构架,如果持久层不选择使用JPA,而是用一些ORM框架(如Hibernate、TopLink)的专用API,那么在将来的某一天一定会为这个选择而说出至尊宝那句“假如上天再给我一个机会…”的至理名言。
       下面是一个简单的Entity,是对一个CMS系统中,关于树状信息目录实体类的定义,包括了一些详细的映射的配置信息。

@Entity
public class NewsDir ...{
    @Id
    @GeneratedValue(strategy = GenerationType.TABLE)
    private Long id;// 主键

    @Column(unique = true, nullable = false, length = 16)
    private String sn;// 目录编号

    private String title; // 目录名称

    @OneToMany(mappedBy = "parent", cascade = javax.persistence.CascadeType.REMOVE)
    private List<NewsDir> children = new java.util.ArrayList<NewsDir>();// 下级目录

    @ManyToOne
    private NewsDir parent;// 父级目录



}
  当然,跟任何其它优秀的技术一样,JPA也不是完美的,在使用的过程中难免都会出这样那样的问题,这就需要我们程序员具有格物致知的本领,在应用中灵活应付这些问题。
  这里例举一个缓迟加载的问题,以上面的新闻目录Entity为例。对于parnet与children这个一对多的双向关联,为了提高系统效率,children默认使用的是缓迟加载的方式。在一些轻量级的构架中,由于脱离了J2EE容器及事务支持,经常会出现Entity脱离了Persitence Context,变成了detach或EntityManager关闭,导致一些我们预想中的一些功能无法正常运行。
  最常见的就是在使用MVC框架的时候,在表示层无法加载需要缓迟加载的数据。比如,在一个基于EasyJWeb的mvc应用中,action中的方法如下:

public Page doList(WebForm form, Module module) ...{
        NewsDirQueryObject ndqo = new NewsDirQueryObject();
        form.toPo(ndqo);
        ndqo.setDel(true);
        IPageList pageList = service.queryDirsByConditions(ndqo);
        CommUtilForTeaec.saveIPageList2WebForm(pageList, form);
        form.addResult("dirPath", this.getDirPath(form));
        return module.findPage("list");
    }
在模板文件中有如下内容:
#foreach($info in ${dir.children})
目录名称:${info.title}
#end

关于业务逻辑层Bean的配置:

 

<aop:config>
                <aop:pointcut id="CmsManage"
            expression="execution(* com.easyjf.cms.service.*.*(..))" />
<aop:advisor advice-ref="cmsManageAdvice"
            pointcut-ref="CmsManage" />            
<tx:advice id="cmsManageAdvice"
        transaction-manager="transactionManager">
        <tx:attributes>
            <tx:method name="get*" propagation="SUPPORTS"
                read-only="true" />
            <tx:method name="query*" propagation="SUPPORTS"
                read-only="true" />
            <tx:method name="*" propagation="REQUIRED" />
        </tx:attributes>
    </tx:advice>
<bean id="cmsManageService"
        class="com.easyjf.cms.service.impl.CmsManageServiceImpl">    
        <property name="newsDirDao" ref="newsDirDao" />
    </bean>
在这里,当mvc层执行到$!info.getChildren()方法的时候,将会用到缓迟加载,由于Spring的事务是配置在service层的,因此在执行service.queryDirsByConditions方法完成后就关闭了事务。因此运行程序就会出现类似下面的错误信息:


2007-03-28 00:39:35,750 ERROR [org.hibernate.LazyInitializationException] - failed to lazily initialize a collection of role: com.easyjf.cms.domain.NewsDir.children, no session or session was closed
org.hibernate.LazyInitializationException: failed to lazily initialize a collection of role: com.easyjf.cms.domain.NewsDir.children, no session or session was closed
 at org.hibernate.collection.AbstractPersistentCollection.throwLazyInitializationException(AbstractPersistentCollection.java:358)
 at org.hibernate.collection.AbstractPersistentCollection.throwLazyInitializationExceptionIfNotConnected(AbstractPersistentCollection.java:350)
 at org.hibernate.collection.AbstractPersistentCollection.readSize(AbstractPersistentCollection.java:97)

  使用其它的mvc如struts、webwork乃至spring mvc都会有这样的问题,问题的核心是在事务启动及结束上,由于我们都习惯于在service层而非mvc配置及使用事务,导致了这样的问题。解决的办法其实很简单,就是把事务的启动放到mvc层,让mvc层的controller来开启事务,而让业务层的方法加入的事务中。比如,在EasyJWeb中,可以通过如下的配置来实现实现在action中开启事务:
  在Spring配置文件中配置EasyJWeb的核心处理器,并把process方法添加到事务中,配置文件如下:

 

<aop:config>
        <aop:pointcut id="easyjwebProcessor"
            expression="execution(* com.easyjf.web.RequestProcessor.process(..))" />
        <aop:advisor advice-ref="txEasyjwebProcessorAdvice"
            pointcut-ref="easyjwebProcessor" />
    </aop:config>
    <tx:advice id="txEasyjwebProcessorAdvice"
        transaction-manager="transactionManager">
        <tx:attributes>
            <tx:method name="*" propagation="REQUIRED" read-only="true" />
        </tx:attributes>
    </tx:advice>
    <bean name="EasyJWeb-Processor" class="com.easyjf.web.core.DefaultRequestProcessor"/>
  只需要这样简单的配置,你会惊奇的发现,所有缓迟加载及其它由Persitence Context无效而引起的问题均解决了。

       关于easyjweb与spring的集成,有兴趣的朋友请参考stef_wu的《在EasyJWeb使用spring容器》一文。



版权声明   给作者写信
本篇文章对您是否有帮助?  投票:         投票结果:     9       1
作者其它文章: 作者全部文章
评论人:andylve 发表时间: Thu Mar 29 12:34:15 CST 2007
好,a good wrod

这个文章共有 1 条评论
主题: 在struts里的实现dtree通用树型结构 上一篇文章
返回文章列表 返回〔开发框架〕
下一篇文章 主题: 研读Spring发布包带的例子petClinic源码之第一部分 POJO


文字广告链接
        自主、快速定制基于JAVA的B/S业务系统          重量级企业在线自定义WEB报表平台
        Excel制表、零代码发布、打印、图表结合——快逸报表,免费、稳定、功能强大的java工具
        技术圈: 关于Java、dotNet、PHP、Ruby、奇客、Web2.0等更多资讯博客精选文章

关于 JR  |  版权声明  |  联系我们 

©2002-2006 JR 版权所有 沪ICP备05019622号