从API中,我可以看到它与代理有关。但是我找不到有关代理的大量信息,也不了解调用session.get
和之间的区别session.load
。有人可以解释一下或将我定向到参考页面吗?
谢谢!!
Answers:
摘自《休眠中的行动》一书。好人读过这个..
通过标识符检索对象以下Hibernate代码段从数据库检索User对象:
User user = (User) session.get(User.class, userID);
get()方法很特殊,因为标识符唯一地标识一个类的单个实例。因此,应用程序通常使用标识符作为持久对象的便捷句柄。按标识符检索可以在检索对象时使用高速缓存,从而避免在对象已被高速缓存的情况下数据库命中。Hibernate还提供了load()方法:
User user = (User) session.load(User.class, userID);
load()方法较旧;由于用户请求,get()已添加到Hibernate的API中。区别不大:
如果load()在缓存或数据库中找不到对象,则会引发异常。load()方法从不返回null。如果找不到对象,则get()方法将返回null。
load()方法可以返回代理而不是实际的持久实例。代理是一个占位符,它在首次访问实物时触发加载。另一方面,get()从不返回代理。在get()和load()之间进行选择很容易:如果确定持久对象存在,并且不存在将被认为是例外,则load()是一个不错的选择。如果不确定是否存在具有给定标识符的持久实例,请使用get()并测试返回值以查看其是否为null。使用load()有进一步的含义:应用程序可以检索对持久实例的有效引用(代理),而无需访问数据库来检索其持久状态。因此,当load()在缓存或数据库中找不到持久对象时,它可能不会引发异常。访问代理后,将在以后引发异常。当然,通过标识符检索对象并不像使用任意查询那样灵活。
好吧,至少在nhibernate中,session.Get(id)将从数据库中加载对象,而session.Load(id)仅为其创建代理对象,而无需离开服务器。就像您的POCO(或POJO :)中其他所有延迟加载的属性一样工作。然后,您可以使用此代理作为对对象本身的引用来创建关系等。
可以将其想像成具有仅保留Id的对象,并在需要时加载其余对象。如果只是传递它来创建关系(如FK),则ID就是您所需要的。
还有一点:
如果在缓存以及数据库中都找不到对象,则Hibernate Session类的get方法将返回null。如果在缓存和数据库上均未找到对象,但load()方法将抛出ObjectNotFoundException,但永远不会返回null。
使用“加载”而不是“获取”的间接结果是,使用版本属性进行乐观锁定可能无法按您期望的那样工作。如果加载只是创建一个代理而不从数据库中读取,则不会加载version属性。仅当/如果您以后引用对象上的属性并触发选择时,才会加载版本。同时,另一个会话可以更新该对象,并且您的会话将没有进行乐观锁检查所需的原始版本-因此,您会话的更新将覆盖另一个会话的更新,而不会发出警告。
这是一种尝试通过使用具有相同标识符的对象进行两次会话来勾勒此场景的尝试。DB中对象的初始版本为10。
Session 1 Session 2
--------- ---------
Load object
Wait a while..
Load object
Modify object property
[triggers db 'select' -
version read as 10]
Commit
[triggers db update,
version modified to 11]
Modify object property
[triggers db 'select' -
version read as 11]
Commit
[triggers db update,
version modified to 12]
我们实际上希望会话1的提交因乐观锁异常而失败,但是它将在这里成功。
使用“ get”而不是“ load”可以解决此问题,因为get将立即发出选择,并且将在正确的时间加载版本号以进行乐观锁检查。
在http://www.mkyong.com/hibernate/different-between-session-get-and-session-load
session.load()中找到了一个很好的解释:
它将始终返回一个“代理”(休眠术语),而不会打数据库。
在Hibernate中,proxy是具有给定标识符值的对象,其属性尚未初始化,它看起来像一个临时的伪对象。
它将始终返回具有给定标识值的代理对象,即使该标识值在数据库中不存在。但是,当您尝试通过从数据库检索代理的属性来初始化代理时,它将使用select语句访问数据库。如果未找到任何行,则将引发ObjectNotFoundException。
session.get():
它总是命中数据库(如果未在缓存中找到)并返回真实对象,该对象代表数据库行,而不是代理。
如果未找到行,则返回null。
load()无法从缓存或数据库中找到对象,将引发异常,并且load()方法从不返回null。
如果找不到对象,则get()方法将返回null。load()方法可能返回一个代理,而不是一个真正的持久实例get()从不返回代理。