您的位置:首页 > 其它

04_传智播客hibernate教程_Session接口及getloadpersist方法 1

2009-07-11 22:06 441 查看
Session 的几个主要方法
1 save,persist 保存数据。在开启事务的前提下,两者没有分别。若没有
开启事务,persist 是不会执行insert语句。
2 delete 删除对象
3 update 更新对象。若更新一条不存在的记录,会出现异常
4 get 根据ID查,会立刻访问数据库
5 Load根据ID查,(返回的是代理,不会立即访问数据库)
6 saveOrUpdate,merge(根据id和version的值来确定save或update)
调用merage你的对象还是托管的(ppt上是这么写的)
7 lock(把对象变成持久对象,但不会同步对象状态)
这节课 对以上方法做了个大概说明,具体的在以后的课回详细讲解。但是既然开了头没搞明白 心里总是很不爽。查了资料,做了点例子结合老师讲的 现总结如下

Save 方法 将瞬时对象转换成持久对象(分配一个id (主键)给对象)。会生成一个insert语句。在session范围内,hibernate可以检测到持久对象的属性的改变,在事务提交时同步到数据库中(再生成一条update 语句,以持久对象的属性的最后改变值为准)。
Persist 方法
• persist() makes a transient instance persistent. However, it does not guarantee that the
identifier value will be assigned to the persistent instance immediately, the assignment might
happen at flush time. persist() also guarantees that it will not execute an INSERT statement
if it is called outside of transaction boundaries. This is useful in long-running conversations with
an extended Session/persistence context.
1,persist把一个瞬态的实例持久化,但是并"不保证"标识符被立刻填入到持久化实例中,标识符的填入可能被推迟到flush的时间。(为了提高效率)
2,persist"保证",当它在一个transaction外部被调用的时候并不触发一个Sql Insert,这个功能是很有用的,当我们通过继承Session/persistence context来封装一个长会话流程的时候(是个什么场合?),一个persist这样的函数是需要的。(为了减少数据库的交互,提高效率)
3,save"不保证"第2条,它要返回标识符,所以它会立即执行Sql insert,不管是不是在transaction内部还是外部
Load && get

1 Be aware that load() will throw an unrecoverable exception if there is no matching database row (检索结果为空的话 会抛出异常 ObjectNotFoundException)
.
2 If the class is mapped with a proxy(?), load() just returns an uninitialized proxy and does not actually hit the database until you invoke a method of the proxy. This is useful if you wish to create an association to an object without actually loading it from the database.
如果加载对象对应的类是final 的话,则不会生成代理对象。否则 returns an uninitialized proxy.只有在对加载对象进行操作的时候,才会hit the database 就是执行select 语句

以下转载自http://developer.51cto.com/art/200906/131944.htm

这次我们来谈一下Hibernate3.2 Session加载数据时get和load方法的区别(Hibernate 3以后的版本就用get()方法取代find()这个方法了),其实这个在网上有很多的论述,可大多语焉不详或经不起实践的推敲,让很多初学者学的满腹疑窦,现在我给大家讲解一下:

1. 对于Hibernate get方法,Hibernate会确认一下该id对应的数据是否存在,首先在session缓存中查找,然后在二级缓存中查找,还没有就查询数据库,数据 库中没有就返回null。这个相对比较简单,也没有太大的争议。主要要说明的一点就是在这个版本中get方法也会查找二级缓存!

2. Hibernate load方法加载实体对象的时候,根据映射文件上类级别的lazy属性的配置(默认为true),分情况讨论:

(1)若为true,则首先在Session缓存中查找,看看该id对应的对象是否存在,不存在则使用延迟加载,返回实体的代理类对象(该代理类为 实体类的子类,由CGLIB动态生成)。等到具体使用该对象(除获取OID以外)的时候,再查询二级缓存和数据库,若仍没发现符合条件的记录,则会抛出一 个ObjectNotFoundException。

(2)若为false,就跟Hibernate get方法查找顺序一样,只是最终若没发现符合条件的记录,则会抛出一个ObjectNotFoundException。

这里get和load有两个重要区别:
如果未能发现符合条件的记录,Hibernate get方法返回null,而load方法会抛出一个ObjectNotFoundException。
load 方法可返回没有加载实体数据的代理类实例,而get方法永远返回有实体数据的对象。(对于load和get方法返回类型:好多书中都说:“get方法永远 只返回实体类”,实际上并不正确,get方法如果在session缓存中找到了该id对应的对象,如果刚好该对象前面是被代理过的,如被load方法使用 过,或者被其他关联对象延迟加载过,那么返回的还是原先的代理对象,而不是实体类对象,如果该代理对象还没有加载实体数据(就是id以外的其他属性数 据),那么它会查询二级缓存或者数据库来加载数据,但是返回的还是代理对象,只不过已经加载了实体数据。)总之对于get和load的根本区别,一句话,hibernate对于load方法认为该数据在数据库中一定存在,可以放心的使用代理来延迟加载,如果在使用过程中发现了问题,只能抛异常;而对于get方法,hibernate一定要获取到真实的数据,否则返回n
转载完毕

//返回代理对象
user = (User) session.load(userClass, id);
System.out.println("the object loaded user of class is " + user.getClass().getName());
//返回代理对象
user = (User) session.get(userClass, id);
System.out.println("the object geted user of class is " + user.getClass().getName());
@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@
//返回对象
user = (User) session.get(userClass, id);
System.out.println("the object geted user of class is " + user.getClass().getName());

//返回对象
user = (User) session.load(userClass, id);
System.out.println("the object loaded user of class is " + user.getClass().getName());
总结 session存在的是什么对象 返回的就是什么对象

代理对象继承于domain对象
load方法 返回的对象不可能是null
1 如果domain对象为fianl, 检索记录为空在执行load方法时会抛出 ObjectNotFoundException
2 如果domain对象不为fianl, 检索记录为空在执行load方法返回代理对象(load 方法可返回没有加载实体数据的代理类实例,除了id属性值。这个值就是load方法的参数id,其他属性都没有值) 在操作检索结果对象抛出ObjectNotFoundException
总之 要么抛异常 要么有结果
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: