无法初始化代理 – 没有会话

我有一个看起来像这样的错误:

无法初始化代理 – 没有会话

我正在使用java,hibernate和spring。 尝试生成PDF文档时会出现此错误,我正在按照接下来的步骤动态生成并存储在数据库中。

  1. 我通过POST方法向应用程序发送了一个请求。 这会动态生成PDF并显示给用户。

  2. 就在那个请求之后我发送另一个请求,但通过ajax请求。 这将生成相同的PDF,但会将其保存在数据库中。

该错误显示由于“无法初始化代理 – 无会话”错误而无法执行查询。

有什么我做错了,从同一个用户会话调用两次相同的方法? 是否会在两个请求完成之前关闭会话?

希望有人能帮我理解发生的事情。

您的问题是hibernate Session仅适用于一个请求。 它在请求开始时打开,最后关闭。 您猜对了答案:在两个请求完成之前,Hibernate会话已关闭。

到底发生了什么? 两个请求期间您的实体对象都存在。 怎么样? 它们存储在HTTP会话中(这是一个不同的东西,称为会话)您没有提供有关您正在使用的框架的大量信息,因此我无法向您提供更多详细信息,但可以肯定您正在使用的框架以某种方式将您的实体保留在HTTP会话中。 这就是框架使您可以轻松地为多个请求使用相同的对象。

当第二个请求的处理开始时,代码试图访问由hibernate懒惰地初始化的某个实体(通常是集合的元素)。 该实体未附加到hibernate会话,因此hibernate在读取之前无法初始化hibernate代理。 您应该在ajax请求处理开始时打开一个会话并将实体重新附加到它。

编辑:

我将尝试简要说明幕后发生的事情。 所有java Web框架都有一个或多个servlet来处理请求。 servlet通过创建一个最终产生响应的新线程(HttpResponse)来处理每个请求(HttpRequest)。 处理每个请求的方法在此线程内执行。

在请求处理开始时,您的应用程序应分配处理所需的资源(事务,Hibernate会话等)。 在处理周期结束时,释放这些资源(提交事务,关闭hibernate会话,释放JDBC连接等)。 这些资源的生命周期可以由您的框架管理,也可以由您的代码完成。

为了支持无状态协议中的应用程序状态为HTTP,我们有了HttpSession对象。 我们(或框架)在HttpSession上放置了在同一客户端的不同请求周期之间保持相关的信息。

在处理第一个请求期间,hibernate从数据库中读取(懒惰)一个实体。 由于延迟初始化,该对象结构的某些部分是hibernate代理对象。 这些对象与创建它们的hibernate会话相关联。

当您尝试处理第二个请求时,框架会从HttpSession对象中的先前请求中查找实体。 然后它试图从懒惰初始化的子实体访问属性,现在是一个hibernate代理对象。 hibernate代理对象是对真实对象的模仿,当有人试图访问其中一个属性时,它将要求其hibernate会话用数据库中的信息填充它。 这是你的hibernate代理尝试做的事情。 但是它的会话在前一个请求处理结束时关闭了,所以现在它没有使用hibernate会话来补充水分(填充真实信息)。

请注意,您可能已在第二个请求开始时打开了一个hibernate会话,但它不知道包含代理对象的实体,因为该实体是由另一个hibernate sesion读取的。 您应该将实体重新附加到新的hibernate会话。

关于如何重新附加分离的实体有很多讨论,但现在最简单的方法是session.update(entity)

希望能帮助到你。