作为EJB3.0的一部分,JPA是一个好东西。其简单的配置方式及强大的默认配置支持,使其可以轻松自由的存在于轻量与重量之间,如果现在您" />

让spring帮助你在MVC层解决JPA的缓迟加载问题

系统 1809 0
<iframe align="center" marginwidth="0" marginheight="0" src="http://www.zealware.com/csdnblog336280.html" frameborder="0" width="336" scrolling="no" height="280"></iframe>

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

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

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

private Stringtitle; // 目录名称

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

@ManyToOne
private NewsDirparent; // 父级目录



}

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

public PagedoList(WebFormform,Modulemodule) ... {
NewsDirQueryObjectndqo
= new NewsDirQueryObject();
form.toPo(ndqo);
ndqo.setDel(
true );
IPageListpageList
= 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] - failedtolazilyinitializeacollectionofrole:com.easyjf.cms.domain.NewsDir.children,nosessionorsessionwasclosed
org.hibernate.LazyInitializationException:failedtolazilyinitializeacollectionofrole:com.easyjf.cms.domain.NewsDir.children,nosessionorsessionwasclosed
atorg.hibernate.collection.AbstractPersistentCollection.throwLazyInitializationException(AbstractPersistentCollection.java:
358 )
atorg.hibernate.collection.AbstractPersistentCollection.throwLazyInitializationExceptionIfNotConnected(AbstractPersistentCollection.java:
350 )
atorg.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容器》 一文。




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

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

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

private Stringtitle; // 目录名称

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

@ManyToOne
private NewsDirparent; // 父级目录



}

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

public PagedoList(WebFormform,Modulemodule) ... {
NewsDirQueryObjectndqo
= new NewsDirQueryObject();
form.toPo(ndqo);
ndqo.setDel(
true );
IPageListpageList
= 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] - failedtolazilyinitializeacollectionofrole:com.easyjf.cms.domain.NewsDir.children,nosessionorsessionwasclosed
org.hibernate.LazyInitializationException:failedtolazilyinitializeacollectionofrole:com.easyjf.cms.domain.NewsDir.children,nosessionorsessionwasclosed
atorg.hibernate.collection.AbstractPersistentCollection.throwLazyInitializationException(AbstractPersistentCollection.java:
358 )
atorg.hibernate.collection.AbstractPersistentCollection.throwLazyInitializationExceptionIfNotConnected(AbstractPersistentCollection.java:
350 )
atorg.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容器》 一文。




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

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

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

private Stringtitle; // 目录名称

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

@ManyToOne
private NewsDirparent; // 父级目录



}

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

public PagedoList(WebFormform,Modulemodule) ... {
NewsDirQueryObjectndqo
= new NewsDirQueryObject();
form.toPo(ndqo);
ndqo.setDel(
true );
IPageListpageList
= 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] - failedtolazilyinitializeacollectionofrole:com.easyjf.cms.domain.NewsDir.children,nosessionorsessionwasclosed
org.hibernate.LazyInitializationException:failedtolazilyinitializeacollectionofrole:com.easyjf.cms.domain.NewsDir.children,nosessionorsessionwasclosed
atorg.hibernate.collection.AbstractPersistentCollection.throwLazyInitializationException(AbstractPersistentCollection.java:
358 )
atorg.hibernate.collection.AbstractPersistentCollection.throwLazyInitializationExceptionIfNotConnected(AbstractPersistentCollection.java:
350 )
atorg.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容器》 一文。




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

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

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

private Stringtitle; // 目录名称

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

@ManyToOne
private NewsDirparent; // 父级目录



}

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

public PagedoList(WebFormform,Modulemodule) ... {
NewsDirQueryObjectndqo
= new NewsDirQueryObject();
form.toPo(ndqo);
ndqo.setDel(
true );
IPageListpageList
= 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] - failedtolazilyinitializeacollectionofrole:com.easyjf.cms.domain.NewsDir.children,nosessionorsessionwasclosed
org.hibernate.LazyInitializationException:failedtolazilyinitializeacollectionofrole:com.easyjf.cms.domain.NewsDir.children,nosessionorsessionwasclosed
atorg.hibernate.collection.AbstractPersistentCollection.throwLazyInitializationException(AbstractPersistentCollection.java:
358 )
atorg.hibernate.collection.AbstractPersistentCollection.throwLazyInitializationExceptionIfNotConnected(AbstractPersistentCollection.java:
350 )
atorg.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容器》 一文。




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

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

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

private Stringtitle; // 目录名称

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

@ManyToOne
private NewsDirparent; // 父级目录



}

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

public PagedoList(WebFormform,Modulemodule) ... {
NewsDirQueryObjectndqo
= new NewsDirQueryObject();
form.toPo(ndqo);
ndqo.setDel(
true );
IPageListpageList
= 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] - failedtolazilyinitializeacollectionofrole:com.easyjf.cms.domain.NewsDir.children,nosessionorsessionwasclosed
org.hibernate.LazyInitializationException:failedtolazilyinitializeacollectionofrole:com.easyjf.cms.domain.NewsDir.children,nosessionorsessionwasclosed
atorg.hibernate.collection.AbstractPersistentCollection.throwLazyInitializationException(AbstractPersistentCollection.java:
358 )
atorg.hibernate.collection.AbstractPersistentCollection.throwLazyInitializationExceptionIfNotConnected(AbstractPersistentCollection.java:
350 )
atorg.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容器》 一文。




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

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

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

private Stringtitle; // 目录名称

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

@ManyToOne
private NewsDirparent; // 父级目录



}

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

public PagedoList(WebFormform,Modulemodule) ... {
NewsDirQueryObjectndqo
= new NewsDirQueryObject();
form.toPo(ndqo);
ndqo.setDel(
true );
IPageListpageList
= 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] - failedtolazilyinitializeacollectionofrole:com.easyjf.cms.domain.NewsDir.children,nosessionorsessionwasclosed
org.hibernate.LazyInitializationException:failedtolazilyinitializeacollectionofrole:com.easyjf.cms.domain.NewsDir.children,nosessionorsessionwasclosed
atorg.hibernate.collection.AbstractPersistentCollection.throwLazyInitializationException(AbstractPersistentCollection.java:
358 )
atorg.hibernate.collection.AbstractPersistentCollection.throwLazyInitializationExceptionIfNotConnected(AbstractPersistentCollection.java:
350 )
atorg.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容器》 一文。




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

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

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

private Stringtitle; // 目录名称

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

@ManyToOne
private NewsDirparent; // 父级目录



}

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

public PagedoList(WebFormform,Modulemodule) ... {
NewsDirQueryObjectndqo
= new NewsDirQueryObject();
form.toPo(ndqo);
ndqo.setDel(
true );
IPageListpageList
= 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] - failedtolazilyinitializeacollectionofrole:com.easyjf.cms.domain.NewsDir.children,nosessionorsessionwasclosed
org.hibernate.LazyInitializationException:failedtolazilyinitializeacollectionofrole:com.easyjf.cms.domain.NewsDir.children,nosessionorsessionwasclosed
atorg.hibernate.collection.AbstractPersistentCollection.throwLazyInitializationException(AbstractPersistentCollection.java:
358 )
atorg.hibernate.collection.AbstractPersistentCollection.throwLazyInitializationExceptionIfNotConnected(AbstractPersistentCollection.java:
350 )
atorg.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容器》 一文。




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

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

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

private Stringtitle; // 目录名称

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

@ManyToOne
private NewsDirparent; // 父级目录



}

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

public PagedoList(WebFormform,Modulemodule) ... {
NewsDirQueryObjectndqo
= new NewsDirQueryObject();
form.toPo(ndqo);
ndqo.setDel(
true );
IPageListpageList
= 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] - failedtolazilyinitializeacollectionofrole:com.easyjf.cms.domain.NewsDir.children,nosessionorsessionwasclosed
org.hibernate.LazyInitializationException:failedtolazilyinitializeacollectionofrole:com.easyjf.cms.domain.NewsDir.children,nosessionorsessionwasclosed
atorg.hibernate.collection.AbstractPersistentCollection.throwLazyInitializationException(AbstractPersistentCollection.java:
358 )
atorg.hibernate.collection.AbstractPersistentCollection.throwLazyInitializationExceptionIfNotConnected(AbstractPersistentCollection.java:
350 )
atorg.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容器》 一文。




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

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

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

private Stringtitle; // 目录名称

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

@ManyToOne
private NewsDirparent; // 父级目录



}

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

public PagedoList(WebFormform,Modulemodule) ... {
NewsDirQueryObjectndqo
= new NewsDirQueryObject();
form.toPo(ndqo);
ndqo.setDel(
true );
IPageListpageList
= 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] - failedtolazilyinitializeacollectionofrole:com.easyjf.cms.domain.NewsDir.children,nosessionorsessionwasclosed
org.hibernate.LazyInitializationException:failedtolazilyinitializeacollectionofrole:com.easyjf.cms.domain.NewsDir.children,nosessionorsessionwasclosed
atorg.hibernate.collection.AbstractPersistentCollection.throwLazyInitializationException(AbstractPersistentCollection.java:
358 )
atorg.hibernate.collection.AbstractPersistentCollection.throwLazyInitializationExceptionIfNotConnected(AbstractPersistentCollection.java:
350 )
atorg.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容器》 一文。




让spring帮助你在MVC层解决JPA的缓迟加载问题


更多文章、技术交流、商务合作、联系博主

微信扫码或搜索:z360901061

微信扫一扫加我为好友

QQ号联系: 360901061

您的支持是博主写作最大的动力,如果您喜欢我的文章,感觉我的文章对您有帮助,请用微信扫描下面二维码支持博主2元、5元、10元、20元等您想捐的金额吧,狠狠点击下面给点支持吧,站长非常感激您!手机微信长按不能支付解决办法:请将微信支付二维码保存到相册,切换到微信,然后点击微信右上角扫一扫功能,选择支付二维码完成支付。

【本文对您有帮助就好】

您的支持是博主写作最大的动力,如果您喜欢我的文章,感觉我的文章对您有帮助,请用微信扫描上面二维码支持博主2元、5元、10元、自定义金额等您想捐的金额吧,站长会非常 感谢您的哦!!!

发表我的评论
最新评论 总共0条评论