引用 http://blog.csdn.net/chinarefers/archive/2005/01/20/261252.aspx ywang的提问:
unsaved-value是表示一个对象是新的还是旧的,如果unsaved-value=none 那么就是新的,就会被insert到数据库中,如果unsaved-value=any 就是说明对象是从数据库中load的,被update到数据库中。 我的问题是:unsaved-value是由我们来强制说明这个对象是新的还是旧的,那如果我把一个对象的unsaved-value设置为any,那我要new 一个对象,把他save到数据库中,怎么做呢?我感觉这不是矛盾了吗?主要是我们在写配置文件的时候怎么能说一个对象就一定是new的还是load的? 我查过过去的一些帖子,都是讲到unsaved-value是什么表示什么意思,我还不是不懂,希望大家教我。 robbin的回答: 当你显式的使用session.save()或者session.update()操作一个对象的时候,实际上是用不到unsaved-value的。某些情况下(父子表关联保存),当你在程序中并没有显式的使用save或者update一个持久对象,那么Hibernate需要判断被操作的对象究竟是一个已经持久化过的持久对象,是一个尚未被持久化过的内存临时对象。例如: Session session = ...;Transaction tx = ...;Parent parent = (Parent) session.load(Parent.class, id);Child child = new Child();child.setParent(parent);child.setName("sun");parent.addChild(child);s.update(parent);s.flush();tx.commit();s.close(); 因此我们需要暗示一下Hibernate,究竟child对象应该对它自动save还是update。在上例中,显然我们应该暗示Hibernate对child自动save,而不是自动update。那么Hibernate如何判断究竟对child是save还是update呢?它会取一下child的主键属性 child.getId() ,这里假设id是 java.lang.Integer类型的。如果取到的Id值和hbm映射文件中指定的unsave-value相等,那么Hibernate认为child是新的内存临时对象,发送save,如果不相等,那么Hibernate认为child是已经持久过的对象,发送update。 unsaved-value="null" (默认情况,适用于大多数对象类型主键 Integer/Long/String/...) 当Hibernate取一下child的Id,取出来的是null(在上例中肯定取出来的是null),和unsaved-value设定值相等,发送save(child) 当Hibernate取一下child的id,取出来的不是null,那么和unsaved-value设定值不相等,发送update(child) 例如下面的情况: Session session = ...;Transaction tx = ...;Parent parent = (Parent) session.load(Parent.class, id);Child child = (Child) session.load(Child.class, childId);child.setParent(parent);child.setName("sun");parent.addChild(child);s.update(parent);s.flush();tx.commit();s.close(); BTW: parent对象不需要操心,因为程序显式的对parent有load操作和update的操作,不需要Hibernate自己来判断究竟是save还是update了。我们要注意的只是child对象的操作。另外unsaved-value是定义在Child类的主键属性中的。 <class name="Child" table="child"><id column="id" name="id" type="integer" unsaved-value="null"> <generator class="identity"/></id>...</class> unsaved-null="0" unsaved-value="none"和 unsaved-value="any" 主主要用在主键属性不是通过Hibernate生成,而是程序自己setId()的时候。 在这里多说一句,强烈建议使用Hibernate的id generator,或者你可以自己扩展Hibernate的id generator,特别注意不要使用有实际含义的字段当做主键来用!例如用户类User,很多人喜欢用用户登陆名称做为主键,这是一个很不好的习惯,当用户类和其他实体类有关联关系的时候,万一你需要修改用户登陆名称,一改就需要改好几张表中的数据。偶合性太高,而如果你使用无业务意义的id generator,那么修改用户名称,就只修改user表就行了。 由这个问题引申出来,如果你严格按照这个原则来设计数据库,那么你基本上是用不到手工来setId()的,你用Hibernate的id generator就OK了。因此你也不需要了解当 unsaved-value="none"和 unsaved-value="any" 究竟有什么含义了。如果你非要用assigned不可,那么继续解释一下: unsaved-value="none" 的时候,由于不论主键属性为任何值,都不可能为none,因此Hibernate总是对child对象发送update(child) unsaved-value="any" 的时候,由于不论主键属性为任何值,都肯定为any,因此Hibernate总是对child对象发送save(child) 大多数情况下,你可以避免使用assigned,只有当你使用复合主键的时候不得不手工setId(),这时候需要你自己考虑究竟怎么设置unsaved-value了,根据你自己的需要来定。 BTW: Gavin King强烈不建议使用composite-id,强烈建议使用UserType。 因此,如果你在系统设计的时候,遵循如下原则: 1、使用Hibernate的id generator来生成无业务意义的主键,不使用有业务含义的字段做主键,不使用assigned。 2、使用对象类型(String/Integer/Long/...)来做主键,而不使用基础类型(int/long/...)做主键 3、不使用composite-id来处理复合主键的情况,而使用UserType来处理该种情况。 那么你永远用的是unsaved-value="null" ,不可能用到any/none/..了。
Trackback: http://tb.blog.csdn.net/TrackBack.aspx?PostId=261252
|
联系客服