![](/img/trans.png)
[英]Hibernate with Kotlin: @ManyToOne(fetch = FetchType.LAZY)
[英]Fetch data from table using hibernate with fetch=FetchType.LAZY
我正在尝试从表中获取所有用户的电子邮件。 实体用户:
@Entity
@Table(name = "tbl_User")
public class User {
@Expose
@Id
@GeneratedValue
@Column(name = "id")
private Long id;
.....
@OneToMany(mappedBy = "user", fetch = FetchType.LAZY)
List<CommunicationAddress> communicationAddresses = new ArrayList<CommunicationAddress>();
.....
}
在服务中,我正在吸引用户并尝试查看电子邮件:
User user = userDAO.getUserById(id);
if (user == null) {
throw new Exception("User not found");
} else {
List<Email> addresses = user.getCommunicationAddresses();
}
但是我收到了下一个例外:
org.hibernate.LazyInitializationException: could not initialize proxy - no Session
at org.hibernate.collection.internal.AbstractPersistentCollection.withTemporarySessionIfNeeded(AbstractPersistentCollection.java:186)
at org.hibernate.collection.internal.AbstractPersistentCollection.readSize(AbstractPersistentCollection.java:137)
at org.hibernate.collection.internal.PersistentBag.isEmpty(PersistentBag.java:249)
获取用户的方法:
@Transactional
@Override
public User getUserById(Long userId) {
Criteria criteria = sessionFactory.getCurrentSession().createCriteria(User.class);
criteria.add(Restrictions.eq("id", userId));
return (User) criteria.uniqueResult();
}
我了解使用条件获取用户时必须获取通讯地址。 谢谢大家
看来您的服务方法未使用@Transactional
注释。 因此,在调用userDAO.getUserById(id);
,不再有交易。 这意味着您不能访问未在事务内部访问/未预取的已加载实体的任何延迟加载属性,而不会遇到LazyInitializationException。
因此,您可以考虑用EAGER
抓取替换LAZY
(这主要取决于您面临的用例),或者应该使用@Transactional
注释Service方法。
我强烈建议注释您的服务方法(而不是DAO方法),因为只有这样您才能与多个实体交互时建立有意义的事务边界。
另外,如果您使用延迟加载,则必须注意离开服务层后可能会发生这种异常的可能性,例如在渲染视图时(假设您以某种方式呈现数据)。
要触发即时加载惰性关联(称为“ 动态关联获取 ”),请在getUserById
添加以下行:
criteria.setFetchMode("communicationAddresses", FetchMoode.EAGER);
但是,如果您以这种特定方法进行操作,我想知道为什么您仍然坚持延迟加载吗?
声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.