博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
让spring帮助你在MVC层解决JPA的缓迟加载问题
阅读量:4192 次
发布时间:2019-05-26

本文共 3925 字,大约阅读时间需要 13 分钟。

       作为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的一文。

Trackback: http://tb.blog.csdn.net/TrackBack.aspx?PostId=1543463

你可能感兴趣的文章
一个开源的IoC采集服务器体系结构设计
查看>>
“人民币找零”问题的贪婪法解决算法
查看>>
“装箱”问题的贪婪法解决算法
查看>>
“马的遍历”问题的贪婪法解决算法
查看>>
国际象棋“皇后”问题的回溯算法
查看>>
长整数的基本操作
查看>>
用长整数求取普通整数的阶乘
查看>>
竞赛选手问题的解答算法
查看>>
[收藏]伟大架构师的秘密
查看>>
Verilog与C++的类比
查看>>
为 LaTeX 添加英文 TrueType 字体
查看>>
《Word排版艺术》读后感——兼谈与LaTeX的比较
查看>>
while (n-- > 0) 与 while (--n >= 0)
查看>>
LaTeX 与字体
查看>>
LaTeX 常用功能
查看>>
变长参数的 Tracer
查看>>
Linux 下配置 802.1X
查看>>
书籍的基本结构, in XML & LaTeX
查看>>
ECOIII專欄,第3集
查看>>
ECO技術和高雄/台中ECO/AJAX技術研討會
查看>>