spring在MVC层解决JPA的缓迟加载问题

作为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无效而引起的问题均解决了。

时间: 2024-10-01 10:49:51

spring在MVC层解决JPA的缓迟加载问题的相关文章

spring 单例多例,默认,懒加载,初始化调用

(1)spring 默认是单例, 单例情况下:所有线程对于同一个类共同拥有一个对象,此时如果在类里面创建一个类变量如下 由于所有线程共享一个类对象,所以也共享一个类变量,每次请求都会增加 @Controller @RequestMapping("/user") public class User { private int i=0;//类变量 @RequestMapping("/list.shtm") public String list() { i++; Syst

解决tableView中cell动态加载控件的重用问题

解决tableView中cell动态加载控件的重用问题 tableView的cell,有时候需要在运行时取得对应的数据后才能够动态的创建该cell中的控件并加载到该cell中,此时,你一定会遇到重用问题,即使你能做到该cell只根据数值加载了一回控件,你也没法保证不出现重用问题:) 效果(请注意查看,移动下面的格子时,上面出现了重用的问题) 源码: YXCell.h // // YXCell.h // YXTableView // // Copyright (c) 2014年 Y.X. All

解决ArcGIS API for Silverlight 加载地图的内外网访问问题

原文:解决ArcGIS API for Silverlight 加载地图的内外网访问问题 先上一个类,如下: public class BaseClass { public static string getFullUri(string oldUriString) { string newUriString = oldUriString; //处理相对地址============================================================ if (newUri

如何解决estjs中异步数据加载失败问题(加载数据超时导致数据加载失败),或延长extjs异步数据加载时间?

问题描述 如何解决estjs中异步数据加载失败问题(加载数据超时导致数据加载失败),或延长extjs异步数据加载时间? 问题补充:lizhi92574 写道 解决方案 Ext.data.Connection.prototype.timeout='9000';设置ajax请求时间默认30秒解决方案二:对解决方案三:你加载多大的数据居然超时了?

解决Windows 10下无法加载桌面背景的问题

1.重启电脑了,我们可以尝试重启电脑看看可以解决不,或者是设置桌面壁纸试一下了,具体:右击桌面空白处,然后点击"个性化"选项,进入后就可以设置壁纸了. 2.使用WIN+R组合键调出运行,输入regedit,调出注册表编辑器(注册表修改前最好是备份了,这个在注册表编辑器上面的菜单可以直接导出备份了): 2.定位到\HEKY_LOCAL_MACHINE\SOFTWARE\MICROSOFT\WINDOWS NT\CURRENTVERSION\WINLOGON,在右侧找到shell,双击打开

解决android studio 3.0 加载项目过慢问题--maven仓库选择

今天用android studio 3.0打开项目时发现一直在谷歌的maven仓库加载 卡到这不动了,看了下maven仓库的配置发现: buildscript { repositories { jcenter() maven { url 'https://maven.google.com' name 'Google' } google() } dependencies { classpath 'com.android.tools.build:gradle:3.0.0' // NOTE: Do n

spring mvc-spring MVC redirect跳转后网址多加了一堆参数

问题描述 spring MVC redirect跳转后网址多加了一堆参数 spring mvc页面转向后,跳转后的地址变为如何去掉?后面的参数转发的代码如下:return ""redirect:/fund/tzyy_success.html""; 转发url对应的方法 @RequestMapping(value = ""/fund/tzyy_success.html"") public String anonymous(Htt

Spring Cloud实战小贴士:Ribbon的饥饿加载(eager-load)模式

2017年架构师最重要的48个小时 | 8折倒计时 我们在使用Spring Cloud的Ribbon或Feign来实现服务调用的时候,如果我们的机器或网络环境等原因不是很好的话,有时候会发现这样一个问题:我们服务消费方调用服务提供方接口的时候,第一次请求经常会超时,而之后的调用就没有问题了.下面我们就来说说造成这个问题的原因,以及如何解决的方法. 问题原因 造成第一次服务调用出现失败的原因主要是Ribbon进行客户端负载均衡的Client并不是在服务启动的时候就初始化好的,而是在调用的时候才会去

Spring Cloud实战小贴士:Zuul的饥饿加载(eager-load)使用

上一篇我们介绍了如何使用Ribbon的earger-load配置加速Spring Cloud中对服务接口的第一次调用.可是这样只是解决了内部服务间的调用,另外一个问题依然经常困扰我们,那就是网关到内部服务的访问.由于Spring Cloud Zuul的路由转发也是通过Ribbon实现负载均衡的,所以它也会存在第一次调时比较慢的情况.那么这个时候我们要如何设置呢? Zuul中的Eager Load配置 在Spring Cloud Zuul中也提供了一个配置参数来实现earger-load,具体如下