org.hibernate.LazyInitializationException:无法初始化代理 - 没有会话。 Spring + Hibernate + HSQLDB

Posted

技术标签:

【中文标题】org.hibernate.LazyInitializationException:无法初始化代理 - 没有会话。 Spring + Hibernate + HSQLDB【英文标题】:org.hibernate.LazyInitializationException: could not initialize proxy - no Session. Spring + Hibernate + HSQLDB 【发布时间】:2014-04-28 14:23:32 【问题描述】:

我在使用 Spring + Hibernate + HSQLDB 配置时遇到了一些问题。 我收到:

org.springframework.web.util.NestedServletException: Request processing failed; nested exception is org.hibernate.LazyInitializationException: could not initialize proxy - no Session
org.springframework.web.servlet.FrameworkServlet.processRequest(FrameworkServlet.java:894)
org.springframework.web.servlet.FrameworkServlet.doGet(FrameworkServlet.java:778)
javax.servlet.http.HttpServlet.service(HttpServlet.java:621)
javax.servlet.http.HttpServlet.service(HttpServlet.java:728)
root cause

org.hibernate.LazyInitializationException: could not initialize proxy - no Session
org.hibernate.proxy.AbstractLazyInitializer.initialize(AbstractLazyInitializer.java:167)
org.hibernate.proxy.AbstractLazyInitializer.getImplementation(AbstractLazyInitializer.java:215)
org.hibernate.proxy.pojo.javassist.JavassistLazyInitializer.invoke(JavassistLazyInitializer.java:190)
org.siwoz.dao.model.Person_$$_javassist_0.toString(Person_$$_javassist_0.java)
org.siwoz.controller.IndexController.index(IndexController.java:31)
sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:62)
sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
java.lang.reflect.Method.invoke(Method.java:483)
org.springframework.web.method.support.InvocableHandlerMethod.invoke(InvocableHandlerMethod.java:212)
org.springframework.web.method.support.InvocableHandlerMethod.invokeForRequest(InvocableHandlerMethod.java:126)
org.springframework.web.servlet.mvc.method.annotation.ServletInvocableHandlerMethod.invokeAndHandle(ServletInvocableHandlerMethod.java:96)
org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerAdapter.invokeHandlerMethod(RequestMappingHandlerAdapter.java:617)
org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerAdapter.handleInternal(RequestMappingHandlerAdapter.java:578)
org.springframework.web.servlet.mvc.method.AbstractHandlerMethodAdapter.handle(AbstractHandlerMethodAdapter.java:80)
org.springframework.web.servlet.DispatcherServlet.doDispatch(DispatcherServlet.java:900)
org.springframework.web.servlet.DispatcherServlet.doService(DispatcherServlet.java:827)
org.springframework.web.servlet.FrameworkServlet.processRequest(FrameworkServlet.java:882)
org.springframework.web.servlet.FrameworkServlet.doGet(FrameworkServlet.java:778)
javax.servlet.http.HttpServlet.service(HttpServlet.java:621)
javax.servlet.http.HttpServlet.service(HttpServlet.java:728)

我尝试使用 OpenSessionInViewInterceptor(用于休眠 3),但后来我有:

org.springframework.web.util.NestedServletException: Request processing failed; nested exception is java.lang.IllegalArgumentException: Resource must not be null
org.springframework.web.servlet.FrameworkServlet.processRequest(FrameworkServlet.java:894)
org.springframework.web.servlet.FrameworkServlet.doGet(FrameworkServlet.java:778)
javax.servlet.http.HttpServlet.service(HttpServlet.java:621)
javax.servlet.http.HttpServlet.service(HttpServlet.java:728)
root cause

java.lang.IllegalArgumentException: Resource must not be null
org.springframework.util.Assert.notNull(Assert.java:112)
org.springframework.transaction.support.TransactionSynchronizationUtils.unwrapResourceIfNecessary(TransactionSynchronizationUtils.java:62)
org.springframework.transaction.support.TransactionSynchronizationManager.hasResource(TransactionSynchronizationManager.java:124)
org.springframework.orm.hibernate3.support.OpenSessionInViewInterceptor.preHandle(OpenSessionInViewInterceptor.java:143)
org.springframework.web.servlet.handler.WebRequestHandlerInterceptorAdapter.preHandle(WebRequestHandlerInterceptorAdapter.java:54)
org.springframework.web.servlet.DispatcherServlet.doDispatch(DispatcherServlet.java:891)
org.springframework.web.servlet.DispatcherServlet.doService(DispatcherServlet.java:827)
org.springframework.web.servlet.FrameworkServlet.processRequest(FrameworkServlet.java:882)
org.springframework.web.servlet.FrameworkServlet.doGet(FrameworkServlet.java:778)
javax.servlet.http.HttpServlet.service(HttpServlet.java:621)
javax.servlet.http.HttpServlet.service(HttpServlet.java:728)

这是我的 cfg 文件:

@Configuration
@EnableTransactionManagement
@PropertySource( "classpath:db/db.properties" )
@ComponentScan( "my_package_name" )
public class PersistenceConfig 

@Autowired
Environment env;

@Bean
public AnnotationSessionFactoryBean sessionFactory() 
    AnnotationSessionFactoryBean sessionFactory = new AnnotationSessionFactoryBean();
    sessionFactory.setDataSource(restDataSource());
    sessionFactory
            .setPackagesToScan(new String[]  "my_package_name" );
    sessionFactory.setHibernateProperties(hibernateProperties());
    try 
        sessionFactory.afterPropertiesSet();
     catch (Exception e) 
        // TODO Auto-generated catch block
        e.printStackTrace();
    
    return sessionFactory;


@Bean
public HibernateTemplate hibernateTemplate() 
    return new HibernateTemplate(sessionFactory().getObject());


@Bean
public DataSource restDataSource() 
    BasicDataSource dataSource = new BasicDataSource();
    dataSource.setDriverClassName(env.getProperty("jdbc.driverClassName"));
    dataSource.setUrl(env.getProperty("jdbc.url"));
    dataSource.setUsername(env.getProperty("jdbc.user"));
    return dataSource;


@Bean
public HibernateTransactionManager transactionManager() 
    HibernateTransactionManager txManager = new HibernateTransactionManager();
    txManager.setSessionFactory(sessionFactory().getObject());
    return txManager;


@Bean
public HibernateExceptionTranslator hibernateExceptionTranslator() 
    return new HibernateExceptionTranslator();


Properties hibernateProperties() 
    return new Properties() 
        private static final long serialVersionUID = 8249579451140862991L;
        
            setProperty("hibernate.connection.pool_size",
                    env.getProperty("hibernate.conn.pool"));
            setProperty("hibernate.hbm2ddl.auto",
                    env.getProperty("hibernate.hbm2ddl.auto"));
            setProperty("hibernate.dialect",
                    env.getProperty("hibernate.dialect"));
            setProperty("hibernate.globally_quoted_identifiers", "true");
            setProperty(
                    "hibernate.globally_quoted_identifiers",
                    env.getProperty("hibernate.cache.use_second_level_cache"));
            setProperty("hibernate.globally_quoted_identifiers",
                    env.getProperty("hibernate.cache.use_query_cache"));
        
    ;


我有一个抽象存储库,这是一段代码:

public abstract class AbstractRepository<T> 

@Autowired
protected HibernateTemplate hibernateTemplate;

public abstract T getById(long id);

这是我的 PersonRepository:

@Repository("personRepository")
public class PersonRepository extends AbstractRepository<Person> 

@Override
     public Person getById(long id) 
    return hibernateTemplate.load(Person.class, id);

我仅使用注释配置(无 xmls)并从 *.sql 文件加载我的 db 方案和数据 - 我可以在 hsql 管理器中查看方案和数据。 调用 getById() 时发生异常。我还在 sessionFactory.getObject().getAllClassMetadata() 中检查了我的实体 - 它已加载。

春季 - 3.1 休眠 - 3.6.10.Final 雄猫 - 7 HSQLDB - 2.3.2 我认为有一个愚蠢的错误,但我看不到它。 有任何想法吗?非常感谢!

编辑: 我在我的 Tomcat 控制台中发现了奇怪的错误:

[ERROR] [localhost-startStop-1 06:06:19] (SchemaExport.java:create:386) Unsucces
sful: create table `Address` (`id` bigint not null auto_increment, `city` varcha
r(255), `street` varchar(255), primary key (`id`))
[ERROR] [localhost-startStop-1 06:06:19] (SchemaExport.java:create:387) unexpect
ed token:  required: (
[ERROR] [localhost-startStop-1 06:06:19] (SchemaExport.java:create:386) Unsucces
sful: create table `Person` (`id` bigint not null auto_increment, `name` varchar
(255), `pass` varchar(255), `pesel` varchar(255), `surname` varchar(255), primar
y key (`id`))
[ERROR] [localhost-startStop-1 06:06:19] (SchemaExport.java:create:387) unexpect
ed token:  required: (
[ERROR] [localhost-startStop-1 06:06:19] (SchemaExport.java:create:386) Unsucces
sful: create table `Address` (`id` bigint not null auto_increment, `city` varcha
r(255), `street` varchar(255), primary key (`id`))
[ERROR] [localhost-startStop-1 06:06:19] (SchemaExport.java:create:387) unexpect
ed token:  required: (
[ERROR] [localhost-startStop-1 06:06:19] (SchemaExport.java:create:386) Unsucces
sful: create table `Person` (`id` bigint not null auto_increment, `name` varchar
(255), `pass` varchar(255), `pesel` varchar(255), `surname` varchar(255), primar
y key (`id`))
[ERROR] [localhost-startStop-1 06:06:19] (SchemaExport.java:create:387) unexpect
ed token:  required: (

我不知道错误可能出在哪里,我也在添加一段 db_scheme:

CREATE TABLE Address (
id INTEGER GENERATED BY DEFAULT AS IDENTITY primary key,
city varchar(64),
street varchar(64)
);

在 hsqldb gui 中,当我删除/添加方案和数据时,没有错误。

【问题讨论】:

试试get而不是load,后者总是创建一个代理而不是去数据库。只有当您想要访问实际基础记录的属性时,它才会进入数据库。此外,您不应再使用HibernateTemplate,而应使用普通的SessionFactory,使用sessionFactory.getCurrentSession 来访问底层会话。 (这也会告诉您何时没有正确的交易设置)。 是的,我阅读了有关 HibernateTemplate 的信息,我只是想尝试所有可能的解决方案 :) 好吧,我照你说的做了:Session session = sessionFactory.getCurrentSession(); return (Person) session.get(Person.class, 1L);并有以下例外:pastebin.com/sFZiygEp 显然你的映射配置错误。 【参考方案1】:

你需要使用@Transactional注解

试试这样的:

@Repository("personRepository")
@Transactional
public class PersonRepository extends AbstractRepository<Person> 

@Override
     public Person getById(long id) 
    return hibernateTemplate.load(Person.class, id);

我希望这会有所帮助 安杰洛

【讨论】:

我很抱歉之前没有提到 - 我在服务级别使用@Transactional。我的请求链如下所示:JSP(用户进入页面)-控制器-PersonService-PersonRepository。无论如何感谢您的快速回答(无论如何我都尝试过,但结果是一样的)。【参考方案2】:

谢谢大家的回答。 不幸的是,我没有找到任何解决我的问题的方法。 我创建了最简单的只有 id 的实体,问题是一样的。 我使用了不同的数据库 - mysql - 现在一切都很好。

【讨论】:

可以发布您的映射、模型和休眠配置吗?

以上是关于org.hibernate.LazyInitializationException:无法初始化代理 - 没有会话。 Spring + Hibernate + HSQLDB的主要内容,如果未能解决你的问题,请参考以下文章