1. Hibernate的一个常见的错误
org.hibernate.LazyInitializationException: failed to lazily initialize a collection of role: org.liky.pojo.Person.emails, 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) at org.hibernate.collection.PersistentSet.size(PersistentSet.java:114) at org.liky.test.Test.main(Test.java:25) |
该错误是由于Hibernate提供的一个提高查询性能的技术照成的。
该技术为延迟加载:作用是当查询多个表的数据,如果只用当前的表(person),就只查Person表中的数据,当使用到email中的数据时候,再去查询email表
但是,使用延迟加载时,由于一般在开发中都会将Session在使用后及时关闭,因此在进行第二次查询的时候出现了延迟加载的错误。
处理方法:
1, 在实现类中不关闭Session,设置由系统自动进行的Session关闭操作,但是有可能会同时打开过多的连接,导致连接池中没有可用连接,而使其他用户等待,并且经常出现在页面上打开连接的情况,安全性不好。
2, 将延迟加载的设置取消
3, 可以修改hbm.xml文件,将lazy延迟加载的设置取消。
<!-- 表示在Person类中有一个名为emails的set集合,该集合中保存的数据对应email表 --> <set name="emails" table="email" lazy="false"> <!-- email表中有一个字段pid与person表关联 --> <key column="pid"></key> <!-- set集合中保存的数据类型为String ,与email表中的email字段对应 --> <element type="java.lang.String" column="email"></element> </set> |
但是,该方法有个缺陷:当表的关系比较复杂时,经常会出现一次查询了好几百条Sql语句的情况,因此,不建议使用。
3.由开发人员在需要的时候,手工调用getEmails.size();
也可以调用别的方法,目的就是导致再次查询email表中的数据