`
icyheart
  • 浏览: 781289 次
  • 性别: Icon_minigender_1
  • 来自: 北京
社区版块
存档分类
最新评论

Hibernate的unsaved-value

阅读更多

Hibernate 中的一些细节应当注意,这里讨论下hibernate的unsaved-value:
先将这个示例说说:

create table category ( 
catid char(10) not null, 
name varchar(80) null, 
descn varchar(255) null, 
constraint pk_category primary key (catid) 
) type=InnoDB; 

create table product ( 
productid char(10) not null, 
category char(10) not null, 
name varchar(80) null, 
descn varchar(255) null, 
constraint pk_product primary key (productid), 
constraint fk_product_1 foreign key (category) references category (catid) 
) type=InnoDB; 

 

 Category 和 Product 的关系是一对多的关系。
Category.java 和 Product.java 就是 POJO。

Category.hbm.xml: 
    <class name="Category" table="category"> 
        <id name="catid" column="catid" type="java.lang.String" > 
            <generator class="assigned"/> 
        </id> 

        <property name="name" column="name" type="java.lang.String" /> 
        <property name="descn" column="descn" type="java.lang.String" /> 
        
        <set name="products" table="product" inverse="true" cascade="all"> 
            <key column="category"/> 
            <one-to-many class="Product"/> 
        </set> 
    </class> 

Product.hbm.xml: 
    <class name="Product" table="product"> 
        <id name="productid" column="productid" type="java.lang.String" > 
            <generator class="assigned"/> 
        </id> 

        <property name="name" column="name" type="java.lang.String" /> 
        <property name="descn" column="descn" type="java.lang.String" /> 
        
        <many-to-one name="category" 
                     column="category" 
                     class="Category"/> 
        
    </class> 

 

测试的类: 
protected void setUp() throws Exception { 
sessionFactory = new Configuration().configure().buildSessionFactory(); 
session = sessionFactory.openSession(); 
} 

protected void tearDown() throws Exception { 
session.close(); 
sessionFactory.close(); 
} 

public void testSave() throws Exception { 
Category cat = new Category(); 
cat.setCatid("FISH"); 
cat.setName("Fish"); 
cat.setDescn("<image src=\"../images/fish_icon.gif\"><font size=\"5\" color=\"blue\"> Fish</font>"); 

Product pro = new Product(); 
pro.setProductid("K9-BD-01"); 
pro.setName("Bulldog"); 
pro.setDescn("<image src=\"../images/dog2.gif\">Friendly dog from England"); 
pro.setCategory(cat); 
cat.getProducts().add(pro); 


Transaction tx= session.beginTransaction(); 
session.save(cat); 
tx.commit(); 
} 

 

结果就出现摘要中所说的错误: 
Hibernate: insert into category (name, descn, catid) values (?, ?, ?) 
Hibernate: update product set name=?, descn=?, category=? where productid=? 
10:07:08,062 ERROR SessionImpl:2399 - Could not synchronize database state with session 

但将两个 *.hbm.xml 文件中的 
<id name="catid" column="catid" type="java.lang.String"> 改为 
<id name="catid" column="catid" type="java.lang.String" unsaved-value="any"> 
<id name="productid" column="productid" type="java.lang.String"> 改为 
<id name="productid" column="productid" type="java.lang.String" unsaved-value="any"> 
测试正常。 

 以下就要说明这个问题的关键所在了,注意看

从夏昕的 Hibernate 开发指南中可以看到他介绍的 “关于unsaved-value”: 

在非显示数据保存时,Hibernate将根据这个值来判断对象是否需要保存。 
所谓显式保存,是指代码中明确调用session 的save、update、saveOrupdate 方法对对象进行持久化。如:session.save(user); 
而在某些情况下,如映射关系中,Hibernate 根据级联(Cascade)关系对联接类进行保存。此时代码中没有针对级联对象的显示保存语句,需要Hibernate 根据对象当前状态判断是否需要保存到数据库。此时,Hibernate即将根据unsaved-value进行判定。 
首先Hibernate会取出目标对象的id。 
之后,将此值与unsaved-value进行比对,如果相等,则认为目标对象尚未保存,否则,认为对象已经保存,无需再进行保存操作。如:user对象是之前由hibernate从数据库中获取,同时,此user对象的若干个关联对象address 也被加载,此时我们向user 对象新增一个address 对象,此时调用 session.save(user),hibernate会根据unsaved-value判断user对象的数个address 关联对象中,哪些需要执行save操作,而哪些不需要。 
对于我们新加入的address 对象而言,由于其id(Integer 型)尚未赋值,因此为null,与我们设定的unsaved-value(null)相同,因此hibernate将其视为一个未保存对象,将为其生成insert语句并执行。这里可能会产生一个疑问,如果“原有”关联对象发生变动(如user的某个“原有”的address对象的属性发生了变化,所谓“原有”即此address对象已经与user相关联,而不是我们在此过程中为之新增的),此时id值是从数据库中读出,并没有发生改变,自然与unsaved-value(null)也不一样,那么Hibernate是不是就不保存了? 
上面关于PO、VO 的讨论中曾经涉及到数据保存的问题,实际上,这里的“保存”,实际上是“insert”的概念,只是针对新关联对象的加入,而非数据库中原有关联对象的“update”。所谓新关联对象,一般情况下可以理解为未与Session 发生关联的VO。而“原有”关联对象,则是PO。如上面关于PO、VO的讨论中所述:对于save操作而言,如果对象已经与Session相关联(即已经被加入Session的实体容器中),则无需进行具体的操作。因为之后的Session.flush过程中,Hibernate 会对此实体容器中的对象进行遍历,查找出发生变化的实体,生成并执行相应的 update 语句。 

针对上面的例子,当没有设定 unsaved-value="any" 时,也就相当于 unsaved-value="none",不论主键属性为任何值,都不可能为 none,因此 Hibernate 总是对 product 对象发送update(product);unsaved-value="any" 的时候,由于不论主键属性为任何值,都肯定为 any,因此 Hibernate 总是对 product 对象发送 save(product)。 

所以 Hibernate.org.cn 论坛中的 Robbin 建议在系统设计的时候,遵循如下原则: 

1、使用Hibernate的id generator来生成无业务意义的主键,不使用有业务含义的字段做主键,不使用assigned。 

2、使用对象类型(String/Integer/Long/...)来做主键,而不使用基础类型(int/long/...)做主键 

3、不使用composite-id来处理复合主键的情况,而使用UserType来处理该种情况。 

那么你永远用的是unsaved-value="null" ,不可能用到any/none/..了。

 看了以上的东西后明白了许多,我最近也遇到了相同的问题,我在论坛里面也发了相似的贴子

http://www.iteye.com/topic/549468?page=2#1297950

hiberate在查询session的时候是根据id进行查询的,而恰恰在这之前我执行了以下操作

BBO.setMasterId(ABO.getMasterId);

 

这样会根据主键id找到ABO,以至于后来总是update,而不是Insert了,所以,遵循如下原则:

1、使用Hibernate的id generator来生成无业务意义的主键,不使用有业务含义的字段做主键,不使用assigned。 

2、使用对象类型(String/Integer/Long/...)来做主键,而不使用基础类型(int/long/...)做主键 

3、不使用composite-id来处理复合主键的情况,而使用UserType来处理该种情况。

 

分享到:
评论

相关推荐

    Hibernate入门 - 基础配置

    &lt;id name="id" unsaved-value="0"&gt; &lt;property name="name" not-null="true"/&gt; &lt;loader query-ref="person"/&gt; &lt;sql-insert&gt;INSERT INTO PERSON (NAME, ID) VALUES ( UPPER(?), ? )&lt;/sql-insert&gt; &lt;sql-update...

    hibernate-mapping参数详解

    5. `unsaved-value`:这个属性用于标记新创建且尚未保存的实例。它的值通常是某个字段的特殊值,用于区分已保存和未保存的实例。 6. `access`:与`default-access`类似,`access`参数也定义了访问对象属性的策略,...

    Hibernate-开发指南.pdf

    - **关于unsaved-value**: 未保存值的处理。 - **Inverse和Cascade**: 关联关系的维护策略。 - **延迟加载(Lazy Loading)**: 提高应用程序性能的方法。 - **事务管理**: - **基于JDBC的事务管理**: 使用底层...

    hibernate映射文件的详解

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

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

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

    Hibernate version 乐观锁 (xml方式)

    在这个配置中,`&lt;version&gt;`标签用于声明乐观锁的版本字段,`column`属性指定数据库对应的列名,`unsaved-value`属性表示未保存的初始值。 在实际的业务逻辑中,当我们试图更新一个用户时,Hibernate会在更新SQL语句...

    hibernate配置元素例子.txt

    - **`unsaved-value`属性**:未保存状态的值,此处设置为`null`表示当实体处于未保存状态时,该字段的值为`null`。 - **`generator`子元素**:定义主键的生成策略,这里采用`uuid.hex`,意味着使用基于UUID的十六...

    hibernate夏昕.pdf内容全面通俗易懂

    - unsaved-value: 特殊标记,用于标识尚未保存的对象状态。 - Inverse和Cascade: 关联关系中的级联操作。 - 延迟加载(LazyLoading): 减少内存占用和提高性能的技术。 - **事务管理**: - **基于JDBC的事务管理**:...

    《深入浅出Hibernate》读书笔记

    Hibernate采用的是数据版本对比策略,通过比较对象的unsaved-value和目标对象的id来决定是否需要更新。 最后,数据缓存是提高性能的关键。Hibernate提供了两个级别的缓存:内部缓存(一级缓存)和二级缓存。一级...

    Hibernate_DEV_GUIDE

    - **关于unsaved-value**:这是Hibernate中的一种特殊属性设置,用于标识某个属性在对象未保存前的默认值。 - **Inverse和Cascade**:用于控制关联关系中对象的级联操作。 - **延迟加载(LazyLoading)**:这是一种...

    Hibernate开发指南(原版书)

    - **数据访问**:讨论PO(Persistent Object)和VO(Value Object)的区别,以及unsaved-value属性的作用,深入解析Inverse和Cascade机制,解释延迟加载(Lazy Loading)的原理和应用场景。 - **事务管理**:分析...

    hibernate配置详解

    &lt;id name="id" type="string" unsaved-value="null"&gt; &lt;column name="cat_id" sql-type="varchar(20)" not-null="true"/&gt; &lt;column name="NAME" sql-type="varchar(20)" not-null="true"/&gt; &lt;/...

    夏昕-Hibernate+开发指南.pdf

    - **数据访问**:探讨了PO(Plain Old Java Object)和VO(Value Object)的区别,以及unsaved-value、Inverse和Cascade、延迟加载等概念。 - **事务管理**:介绍了基于JDBC和JTA的事务处理方法。 - **锁机制**:讲解...

Global site tag (gtag.js) - Google Analytics