`

收藏:关于unsaved-value 的再问

阅读更多
引用
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();
在上例中,程序并没有显式的session.save(child); 那么Hibernate需要知道child究竟是一个临时对象,还是已经在数据库中有的持久对象。如果child是一个新创建的临时对象(本例中就是这种情况),那么Hibernate应该自动产生session.save(child)这样的操作,如果child是已经在数据库中有的持久对象,那么Hibernate应该自动产生session.update(child)这样的操作。

因此我们需要暗示一下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();
child已经在数据库中有了,是一个持久化的对象,不是新创建的,因此我们希望Hibernate发送update(child),在该例中,Hibernate取一下child.getId(),和unsave-value指定的null比对一下,发现不相等,那么发送update(child)。

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>
如果主键属性不是对象型,而是基本类型,如int/long/double/...,那么你需要指定一个数值型的unsaved-value,例如:

unsaved-null="0"
在此提醒大家,很多人以为对主键属性定义为int/long,比定义为Integer/Long运行效率来得高,认为基本类型不需要进行对象的封装和解构操作,因此喜欢把主键定义为int/long的。但实际上,Hibernate内部总是把主键转换为对象型进行操作的,就算你定义为int/long型的,Hibernate内部也要进行一次对象构造操作,返回给你的时候,还要进行解构操作,效率可能反而低也说不定。因此大家一定要扭转一个观点,在Hibernate中,主键属性定义为基本类型,并不能够比定义为对象型效率来的高,而且也多了很多麻烦,因此建议大家使用对象型的Integer/Long定义主键。

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

分享到:
评论

相关推荐

    Atom-atom-unsaved-changes,在活动编辑器中显示未保存更改的atom包.zip

    这个“Atom-atom-unsaved-changes”是一个专门为Atom编辑器设计的插件,其主要功能是在活动编辑器中突出显示未保存的更改,帮助用户更好地管理他们的工作流程,避免丢失任何重要的编辑。 Atom编辑器的特点包括: 1...

    汇编复习题((Unsaved-310043421380800768)).asd

    汇编复习题((Unsaved-310043421380800768)).asd

    EurekaLog_7.5.0.0_Enterprise

    EurekaLog 7.5 (18-August-2016) 1)..Important: Installation layout was changed. All packages now have version suffix (e.g. EurekaLogCore240.bpl). No files are copied to \bin folder of IDE....

    WiFi_射频电路布线指导(不错)((Unsaved-309873122039401648)).asd

    WiFi_射频电路布线指导(不错)((Unsaved-309873122039401648)).asd

    商丘移动PTN7900L2转L3设备入网及LTE业务割接变更方案((Unsaved-309448813321128224)).asd

    商丘移动PTN7900L2转L3设备入网及LTE业务割接变更方案((Unsaved-309448813321128224)).asd

    git-unsaved:扫描您的项目目录以查找肮脏的git存储库

    $ git-unsaved 扫描您的项目目录以查找肮脏的git存储库。 产品特点 :rocket: 非常快-在扫描目录时提供反馈。 :high_voltage: 清洁输出 :Tokyo_tower: 检查本地提交/分支是否被推送到远程 :eyes: 检测未提交的...

    hibernate常见错误解决方案

    **TransientObjectException: object references an unsaved transient instance** **异常描述:** 此异常表示一个实体引用了另一个尚未保存的瞬态实体。 **解决方法:** - 在保存实体之前先保存所有关联的瞬态...

    Node.js-git-unsaved扫描您的项目目录中的脏git存储库

    从提供的压缩包子文件名称"IonicaBizau-git-unsaved-cd746f5"来看,这可能是作者Ionica Bizau的一个git项目的特定版本。通常,这种命名方式包含了作者的名字、项目名和git的哈希值,这可能是一个git标签或者分支,...

    hibernate-mapping参数详解

    unsaved-value (可选 - 默认为 null) 此属性用于指定一个未保存状态的值。如果实体的某个属性被标记为此值,则表示该实体尚未被持久化。这对于检测实体的状态非常有用,尤其是在 session 中修改实体之前。 #### 5...

    Hibernate主键类型说明和配置手册.doc

    &lt;id name="id" unsaved-value="null"&gt; ``` 2. **sequence**: 此策略适用于支持序列的数据库,如Oracle。你需要指定一个序列名,如下所示: ```xml &lt;id name="id" unsaved-value="0"&gt; ...

    hbm.xml说明

    &lt;id name="id" type="java.lang.Long" column="ID" unsaved-value="null" generator-class="increment"/&gt; ``` #### 四、property标签 `property` 标签用于定义 Java 类的属性与数据库表中列之间的映射关系。它有...

    hibernate映射文件的详解

    如果对象的标识属性值等于`unsaved-value`,Hibernate会认为该实例是新创建的,需要执行`save`操作;否则,执行`update`操作。 4. 主键生成器(Key Generator):用于决定如何生成对象的唯一标识(主键)。`...

    Hibernate 开发指南

    - **关于 unsaved-value**:`unsaved-value` 是 Hibernate 提供的一种特殊值,用于标记尚未保存到数据库的对象。 - **Inverse 和 Cascade**:`inverse` 属性用于控制关联关系的一方是否负责维护另一方的状态。`...

    Hibernate开发指南

    - **关于unsaved-value**:解析了unsaved-value的概念及其应用场景。 - **Inverse和Cascade**:解释了这两个属性的意义以及如何正确使用它们。 - **延迟加载(LazyLoading)**:介绍了延迟加载的概念及其在...

    Digispark Arduino IDE 安装环境

    "Digispark Arduino IDE 安装环境" 指的是关于如何在个人计算机上安装和配置用于编程 Digispark 的 Arduino 开发环境。Digispark 是一种基于 Atmel ATtiny85 微控制器的微型 Arduino 兼容开发板。Arduino IDE 是一个...

    linux 下 带shell功能ftp源代码

    Documents myos Unsaved Document 1 4 send files [usage] :s localfilename [return]: return successfully message,if success, or print error messages error message will be introduced below [eg]...

Global site tag (gtag.js) - Google Analytics