could not initialize proxy - no Session 异常情况探究

本文详细解析了在使用Spring框架与Struts2整合时,遇到的LazyInitializationException异常产生的原因及解决方案。通过对比代码差异,发现关键在于正确配置Struts2与Spring之间的交互,特别是正确管理Action实例与Session范围,从而避免异常发生。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

近期跟着“尚学堂”马士兵的spring教学视频复习了一番,学到45课的时候,得知了这样一个据说经典的异常:

org.hibernate.LazyInitializationException: could not initialize proxy - no Session


此异常产生的原因经过马老师以及我多方的查阅可以总结如下:

由于在数据库获取数据采用的是load的方法 只是一个代理对象  则jsp页面用到相关数据才用发出sql语句对数据库进行查询 但等到jsp页面需要数据发出sql语句 这时session已经关了 ,因此会报出异常。 Spring为此提供了一个拦截器 在jsp页面查询到数据后再将session关闭。


解决方案,个人觉着在这个地方比较有价值的有两个,也正如马老师所说的:

1.将dao层中的load方法改为get方法;

2.在web.xml文件中配置拦截器,主要代码如下:

使用OpenSessionInViewFilter。这种方法是将session交给servlet filter来管理,每当一个请求来之后就会开

启一个session,只有当响应结束后才会关闭。如下:

[html]  view plain copy print ?
  1. <filter-name>hibernateFilter</filter-name>   
  2.      <filter-class>  org.springframework.orm.hibernate3.support.OpenSessionInViewFilter </filter-class>   
  3. </filter   
  4. <filter-mapping>   
  5.      <filter-name>hibernateFilter</filter-name>   
  6.      <url-pattern>/*</url-pattern>   
  7. </filter-mapping>   

此处普遍强调一个,就是这段过滤器需要配置在同在web.xml文件中的struts2过滤器代码之前,否则无效。

详细情况可以谷歌“org.hibernate.LazyInitializationException: could not initialize proxy - no Session”这个异常,文章超多,此处不再复制粘贴。




如果真有那么顺利,我也不用记录这篇博文了。问题就来了,我按照课程及各类高手的博客中讲的在web.xml文件中按要求加入了上面的过滤器,位置也绝对放对了,但是没有生效,四下“谷歌”也没有得出什么线索,虽然也有人遇到这种情况,但都是顺序放错导致的。


没有办法之下,讲马士兵的源码部署起来,居然可以,于是思考是不是jar包版本不一样,所以不行。事实证明和jar包半点关系都没有。

再次猜测是不是项目不是新建的,而是通过拷贝粘贴改改导致的,于是新建项目,然后将代码考入,再次把之前的业务代码拷贝粘贴一遍,依然不行,证明这个猜测也不对。

于是再次把马老师的业务代码考进来,离奇的事就是居然可以。仔细对比,终于找着唯一的不同:

@Component("u")
@Scope("prototype")

public class UserAction extends ActionSupport implements ModelDriven<UserRegisterInfo>{


/**
* 采用DTO对象封装表单请求数据
*/
private static final long serialVersionUID = 1L;

private UserRegisterInfo urinfo = new UserRegisterInfo();//DTO对象


private UserManager userManager;

private List<User> userList;

private User userInfo;



public UserAction(){
ApplicationContext ac = new ClassPathXmlApplicationContext("bean.xml");
userManager = (UserManager) ac.getBean("userManager");
}


/**
* @return the userManager
*/
public UserManager getUserManager() {
return userManager;
}
/**
* @param userManager the userManager to set
*/

public void setUserManager(UserManager userManager) {
this.userManager = userManager;
}



public String execute() throws Exception {
User user = new User();
user.setName(urinfo.getUsername());
user.setPassword(urinfo.getUserpass1());
if(userManager.exists(user)){
return "fail";
}
userManager.add(user);
return "success";
}

/**
* 列出所有用户信息
* @return
* @throws Exception
*/
public String list() throws Exception{
userList = userManager.getAllUserList();
return "list";
}

以上是我配置的action,通过spring注解管理。


<struts>


<package name="registration" extends="struts-default">

<action name="user" class="u">
<!-- com.inta.action.UserAction -->
<result name="success">/registerSuccess.jsp</result>
<result name="fail">/registerFail.jsp</result>
<result name="list">/userList.jsp</result>
<result name="load">/userInfo.jsp</result>
</action>

</package>


</struts>

上面是struts.xml配置文件。




两处着色的区域就是不同点,于是将两处地方前者注释掉,后者改为action的类名路径:com.inta.action.UserAction

再次部署,一切OK。到这里,恍然大悟。其实在马老师第42可时候已经分析过这个问题,只是讲的不是很突出重点,不仔细想就过去了。

问题1在于,SSH框架整合时,struts2是主导,struts2通过和spring的插件struts2-spring-plugin.来管理spring,spirng的作用依然是容器,负责管理各个service层和dao层的bean,但struts2内部本身也有一个容器,会管理action中的各个实例变量,所以即使action上的注解全部去掉,表面上看没有让spring管理起来,其实依然会被实例化成bean,并且是按name方式管理。此时即使在action中为各个实例变量加上注释看起来像用spring管理,其实并不会生效,aciton中什么注释也不用都可以。问题2在于,struts.xml文件那里,如果按我上面的方法所写,意味着user这个anciton指向的是spring管理那个name叫u的bean,但正如马老师课程所说,struts2自己会管理action,如果多余手动注释再去管理action就会产生两个bean,于是问题来了,两个action的bean中指向的session不是同一个,所以按我上面的配置写法,即使通过过滤器扩大了session的会话范围,但是根本没有作用到struts2自己管理的session上来,所以根本不会起作用,只有将struts.xml中的 action用类的完整名,才可以定位到struts2管理的bean上来,过滤器配置才会生效。



补充一点:

如果非要将action的管理全权交给spring来做,事实上工作中也都是这么做的,那可以加回上述注释部分,然后将struts.xml文件中的class改为注释配置的name,此处name为u,因此改为:<action name="user" class="u">

这样一改,action类中的所有实例变量就不会有sturts2来管理,必须由spirng注入,否则就会包上述异常。因此在action中的

public void setUserManager(UserManager userManager) {
this.userManager = userManager;
}

前就需要加注释@resource,这样才完全将action交由spring管理。

<think>好的,我需要解决用户提出的Hibernate懒加载初始化集合时出现的“no Session异常问题。用户遇到了org.hibernate.LazyInitializationException,这通常是因为在Session关闭后尝试访问延迟加载的集合导致的。我应该根据提供的引用资料,整理出可行的解决方案,并按照用户的要求用中文回答,结构清晰,分步骤说明。 首先,看看用户提供的引用内容。引用3提到,当lazy设置为true时,在Session关闭后无法加载关联对象,解决方法包括在Session未关闭前调用Hibernate.initialize()或者访问集合的size()方法。引用4建议在映射文件中设置lazy="false"来关闭延迟加载。引用2提到了其他可能的解决方法,但用户给出的引用中并未详细说明,可能需要结合常见做法补充。 接下来,我需要将这些方法归纳成几个步骤,并详细说明每种方法的适用场景和注意事项。例如,方法一可以是在Session关闭前初始化代理对象,这需要在数据访问层处理。方法二则是调整延迟加载配置,可能影响性能。方法三可能涉及事务管理,确保操作在Session开启状态下进行。另外,可能还需要提到Open Session in View模式,虽然用户提供的引用中没有直接提到,但这是常见的解决方案之一,不过需要谨慎使用,避免事务过长的问题。 还要注意用户提到的代码示例,比如Hibernate.initialize(pojo.getChildren())或调用size()方法。需要解释这些代码的作用和使用方式。同时,设置lazy="false"虽然简单,但可能导致不必要的数据库查询,影响性能,需要权衡利弊。 另外,用户可能需要了解为什么会出现这个异常,所以需要简要解释原因:Hibernate的延迟加载机制在Session关闭后无法继续加载数据,导致异常。这样用户能更好地理解解决方案的逻辑。 最后,确保回答结构清晰,分点说明,每个方法都有步骤和注意事项,可能的问题。同时,生成的相关问题需要与主题相关,帮助用户进一步了解其他可能的问题或优化方法。</think>### Hibernate懒加载异常could not initialize proxy - no Session”解决方案 当使用Hibernate的延迟加载(Lazy Loading)机制时,如果在Session关闭后尝试访问未初始化的关联对象或集合,会抛出`org.hibernate.LazyInitializationException`。以下是具体解决步骤: --- #### **方法一:在Session关闭前手动初始化代理对象** **步骤:** 1. 在数据访问层(DAO层)的查询方法中,通过`Hibernate.initialize()`显式初始化关联对象或集合。 ```java public Pojo getPojoWithChildren(Long id) { Session session = sessionFactory.getCurrentSession(); Pojo pojo = session.get(Pojo.class, id); Hibernate.initialize(pojo.getChildren()); // 强制初始化集合 return pojo; } ``` 2. 或者在业务逻辑中访问集合的某个方法(如`.size()`),触发初始化: ```java pojo.getChildren().size(); // 调用此方法会触发数据加载 ``` **注意事项** 此操作必须在Session未关闭前执行,通常在事务范围内完成[^3]。 --- #### **方法二:调整延迟加载配置** **步骤:** 1. 在实体映射文件(`.hbm.xml`)中,将关联字段的`lazy`属性设为`false`: ```xml <set name="children" lazy="false"> <!-- 关闭延迟加载 --> <key column="parent_id"/> <one-to-many class="Child"/> </set> ``` - 或使用JPA注解(如`@OneToMany(fetch = FetchType.EAGER)`)。 **注意事项** 关闭延迟加载会导致关联数据立即加载,可能增加数据库查询次数,需权衡性能影响[^4]。 --- #### **方法三:延长Session生命周期(事务边界扩展)** **步骤:** 1. 使用Spring的`@Transactional`注解扩展事务范围,确保在业务操作中Session始终开启: ```java @Transactional // 事务覆盖整个方法,Session在此期间保持开启 public void processPojo(Long id) { Pojo pojo = pojoRepository.findById(id); pojo.getChildren().forEach(child -> {...}); // 安全访问集合 } ``` **注意事项** 需确保事务管理配置正确,避免事务过长导致数据库连接池耗尽。 --- #### **方法四:使用Open Session in View模式(谨慎使用)** **步骤:** 1. 在Web应用中,通过过滤器(如Spring的`OpenSessionInViewFilter`)保持Session开启直到视图渲染完成。 ```xml <!-- web.xml配置 --> <filter> <filter-name>OpenSessionInViewFilter</filter-name> <filter-class>org.springframework.orm.hibernate5.support.OpenSessionInViewFilter</filter-class> </filter> ``` **注意事项** 此模式可能掩盖性能问题,且需确保事务与Session边界合理匹配,避免内存泄漏[^3]。 --- ### **根本原因分析** Hibernate的延迟加载依赖`Session`上下文。当Session关闭后,代理对象无法连接数据库加载数据,从而抛出异常。解决方法的核心是**确保数据加载在Session有效期内完成**。 ---
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值