hibernate save,update,saveorupdate方法有什么区别

时间:2022-09-12 17:14:25
save肯定是添加的时候用,update修改时候用,saveorupdate是添加或修改,如果真是这样save和update的存在就没意义了,我们直接saveorupdate就行了。
save在添加用的时候  不会出现索引机制(即遍历目录  效率最高)
update在修改时候要遍历 不存在则会异常
saveorupdate是优先遍历 如果不存在则创建(效率最低)
merge和saveOrUpdate方法区别在于:merge方法是把我们提供的对象转变为托管状态的对象;而saveOrUpdate则是把我们提供的对象变成一个持久化对象;
说的通俗一点就是:saveOrUpdate后的对象会纳入session的管理,对象的状态会跟数据库同步,再次查询该对象会直接从session中取,merge后的对 
象不会纳入session的管理,再次查询该对象还是会从数据库中取。

Session的save()方法使一个临时对象转变为持久化对象。例如以下代码保存一个Customer对象:

Customer customer = new Customer();

customer.setId(new Long(9)); // 为Customer临时对象设置OID是无效的

customer.setName("Tom");

Session session = sessionFactory.openSession();

Transaction tx = session.beginTransaction();

session.save(customer);

session.close();

Session的save()方法完成以下的操作:

(1)把Customer对象加入到缓存中,使它变为持久化对象。

(2)选用映射文件指定的标识符生成器为持久化对象分配惟一的OID。Customer.hbm.xml文件中<id>元素的<generator>子元素指定标识符生成器:

<id name="id" column="ID">

<generator class="increment"/>

</id>

以上程序试图通过setId()方法为Customer临时对象设置OID是无效的。假如起初CUSTOMERS表中没有记录,那么执行完save()方法后,Customer对象的ID为1。如果希望由程序来为Customer对象指定OID,可以调用save()的另一个重载方法:

save(customer, new Long(9));

以上save()方法的第二参数显示指定Customer对象的OID。这种形式的save()方法不推荐使用,尤其在使用代理主键的场合,不应该由程序为持久化对象指定OID。

(3)计划执行一个insert语句,把Customer对象当前的属性值组装到insert语句中:

insert into CUSTOMERS(ID, NAME, ......) values(1, 'Tom', ......);

值得注意的是,save()方法并不立即执行SQL insert语句。只有当Session清理缓存时,才会执行SQL insert语句。如果在save()方法之后,又修改了持久化对象的属性,这会使得Session在清理缓存时,额外执行SQL update语句。以下两段代码尽管都能完成相同的功能,但是左边代码仅执行一条SQL insert语句,而右边代码执行一条SQL insert和一条SQL update语句。左边代码减少了操作数据库的次数,具有更好的运行性能。

Customer customer = new Customer();         Customer customer = new Customer();

// 先设置Customer对象的属性,再保存它       session.save(customer);

customer.setName("Tom");                    // 先保存Customer对象,再修改它的属性

session.save(customer);                    customer.setName("Tom");

transaction.commit();                      transaction.commit();

Hibernate通过持久化对象的OID来维持它和数据库相关记录的对应关系。当Customer对象处于持久化状态时,不允许程序随意修改它的OID,例如:

Customer customer = new Customer();

session.save(customer);

customer.setId(new Long(100)); // 抛出HibernateException

transaction.commit();

以上代码会导致Session在清理缓存时抛出异常。


hibernate save,update,saveorupdate方法有什么区别无论Java对象处于临时状态、持久化状态还是游离状态,应用程序都不应该修改它的OID。因此,比较安全的做法是,在定义持久化类时,把它的setId()方法高为private类型,禁止外部程序访问该方法。


Session的save()方法是用来持久化一个临时对象的。在应用程序中不应该把持久化对象或游离对象传给save()方法。例如以下代码两次调用了Session的save()方法,第二次传给save()方法的Customer对角处于持久化状态,这步操作其它是多余的:

Customer customer = new Customer();

session.save(customer);

customer.setName("Tom");

session.save(customer); // 这步操作是多余的

       transaction.commit();

在例如以下代码把Customer游离对象传给session2的save()方法,session2会把它当做临时对象处理,再次向数据库中插入一条Customer记录:

Customer customer = new Customer();

customer.setName("Tom");

Session session1 = sessionFactory.openSession();

Transaction tx1 = session1.beginTransaction();

session1.save(customer); // 此时Customer对象的ID变为1

tx1.commit();

session1.close(); // 此时Customer对象变为游离对象

Session session2 = sessionFactory.openSession();

Transaction tx2 = session2.beginTransaction();

session2.save(cutomer); // 此时Customer对象的ID变为2

tx2.commit();

session2.close();

尽管以上程序代码能正常运行,但是会导致CUSTOMERS表中有两条代表相同业务实体的记录,因此不符合业务逻辑。

 

hibernate提供了saveOrUpdate的方法来进行数据库的操作。hibernate会根据对象的状态决定是insert还是update,
其根本是通过xml文件中unsaved-value来确定的。如果设置null,系统会根据传入的对象的id的值判断,如果是null,
则表示对象不存在,那么insert;如果不是Null,则表示已经存在,那么update.如果设置为none,那么表示对象不存在,
会始终调用insert;如果设置为any,那么表示对象始终存在,会始终调用update


数据库表主键的知识点:
    Generator 为每个 POJO 的实例提供唯一标识。一般情况,我们使用“native”。
class 表示采用由生成器接口net.sf.hibernate.id.IdentifierGenerator 实现的某个实例,其中包括:


“assigned” 主键由外部程序负责生成,在 save() 之前指定一个。


“hilo” 通过hi/lo 算法实现的主键生成机制,需要额外的数据库表或字段提供高位值来源。


“seqhilo”与hilo 类似,通过hi/lo 算法实现的主键生成机制,需要数据库中的 Sequence,
            适用于支持 Sequence 的数据库,如Oracle。


“increment”主键按数值顺序递增。此方式的实现机制为在当前应用实例中维持一个变量,以保存着当前的最大值,
            之后每次需要生成主键的时候将此值加1作为主键。这种方式可能产生的问题是:不能在集群下使用。


“identity”采用数据库提供的主键生成机制。如DB2、SQL Server、MySQL 中的主键生成机制。


“sequence”采用数据库提供的 sequence 机制生成主键。如 Oralce 中的Sequence。


“native”由 Hibernate 根据使用的数据库自行判断采用 identity、hilo、sequence 其中一种作为主键生成方式。


“uuid.hex”由 Hibernate 基于128 位 UUID 算法 生成16 进制数值(编码后以长度32 的字符串表示)作为主键。


“uuid.string”与uuid.hex 类似,只是生成的主键未进行编码(长度16),不能应用在 PostgreSQL 数据库中。


“foreign”使用另外一个相关联的对象的标识符作为主键。


以下举例:
   1、指定参数的情况:
      <id name="id" unsaved-value="0"> 
        <generator class="sequence">
          <param name="sequence">SEQ_CHILD</param>
        </generator>
      </id>
     使用的是sequence,适合oracle数据库;


2、对于sql server2000中的数据库子增字段,在配置文件使用下列方法实现:
      <id name="id" type="long" unsaved-value="0">
        <column name="id" sql-type="numeric" not-null="true" />
        <generator class="identity" /> 
      </id>
    这里主要是:identity:代表由sql server2000数据库自己提供子增字段.如果要hibernate自己提供,则用increment关键字来实现


3、如果表中的主键用字符串类型:可以用hibernate自己提供的方法实现主键唯一:
     <id name="id" type="string" unsaved-value="null">
         <column name="cid" sql-type="char(32)" not-null="true" />
         <generator class="uuid.hex" />
     </id>
     使用的是uuid.hex: 采用128位的算法来生成一个32位字符串。最通用的一种方式。适用于所有数据库。


重要的知识点:
   1. 如果有部门表,有员工表,员工表中有dep_id,则表部门类和员工类是one-to-many的关系:
      可以使用: ( 在部门类department中使用下列)
      Department类: 
        /** 部门的所有员工   */
       private Set staffs = new TreeSet();
     
       xml的文件:
         <set name="staffs" >
             <key column="dep_id"/>
             <one-to-many class="hbp.sys.data.Staff"/>
         </set>
         如果是list,需要用索引<index> </index>,具体其中的含义,不是很明白.待以后研究
      
    2. 如果部门要有一个负责人,即部门表(tb_department)中有一个字段:staff_id.
        那么表示部门和负责人之间的关系是many-to-one的关系
        Department类: 
         /** 部门负责人id */
       private Staff staff;
    
       xml 文件
        <many-to-one name="staff" class="hbp.sys.data.Staff" column="staff_id"/>

3. 多对多关系,一般我们是做一个中间关联表.我用角色和权限做了个例子,
         Right(id,name)     Role(id,name)   中间表:tb_role_right(role_id,right_id)
         Right类中有一个Role的集合:private Set roles=new TreeSet();
         Role类中也有一个Right的集合:private Set rights=new TreeSet();
         则两者是明显的多对多关系.使用many-to-many来实现;
         xml文件中
         right.hbm.xml:如下:
           <set name="roles" table="tb_role_right" cascade="all">
              <key column="right_id"/>
              <many-to-many column="role_id" class="hbp.sys.data.Role"/>
           </set> 
         role.hbm.xml文件中类似:
           <set name="rights" table="tb_role_right" cascade="all">
             <key column="role_id"/>
             <many-to-many column="right_id" class="hbp.sys.data.Right"/>
           </set>


4. 几个值得注意的问题:
           a)在xml?映射文件中,写类的名字时一定用类的全名:即:包+类名如:(hbp.sys.data.Staff),这个错误使我费了半天劲.:(
           b)我在写实现基本DAO操作时,写了
                session.delete("from Right as right where right.id="+id); 
                程序死活报错,我折腾了半天,跟踪到底,才恍然大悟,hibernate在解析sql语句的时候把
                其中的right,当成了数据库中的右连接("保留字"),唉,这种关键字,不能随便用啊,:)


5. hibernate中HQL语言的查询根据你的sql的不同而返回不同的对象类型.
            如果你使用session.find(String hql)
            一般会返回一个List,如:from Staff staff;返回的是包含所有的员工对象的集合
            如你的hql为:select count(*) from Staff staff;则返回的是一个Integer对象
            如果你使用的hql为:select count(distinct staff.name),count(*) from Staff staff;则返回的是一个Object
            即Object[],需要先把他转换成Object[],然后在取[0],[1].
            这种设计我不知道hibernate是如何处理的,感觉既好也不好.好的是可以使用一个find获得任意查询
            不好在于根据hql来处理返回结果,容易出错.

														
		

hibernate save,update,saveorupdate方法有什么区别的更多相关文章

  1. Hibernate save&lpar;&rpar;、saveOrUpdate&lpar;&rpar;、merge&lpar;&rpar;的区别

    一. update 和 merge的区别 首先在执行更新操作的时候,两者都必须要有id update是直接执行update 语句,执行后状态为持久化状态 而merge则不一样: 1. 如果sessio ...

  2. Hibernate三种状态的区分,以及save&comma;update&comma;saveOrUpdate&comma;merge等的使用 引自http&colon;&sol;&sol;www&period;blogjava&period;net&sol;TiGERTiAN&sol;archive&sol;2008&sol;10&sol;25&sol;236519&period;html

    Hibernate的对象有3种状态,分别为:瞬时态(Transient). 持久态(Persistent).脱管态(Detached).处于持久态的对象也称为PO(Persistence Object ...

  3. Hibernate三种状态的区分,以及save&comma;update&comma;saveOrUpdate&comma;merge等的使用

    Hibernate的对象有3种状态,分别为:瞬时态(Transient). 持久态(Persistent).脱管态(Detached).处于持久态的对象也称为PO(Persistence Object ...

  4. Save Update saveOrUpdate delete

    参考:Hibernate Session的saveOrUpdate()方法 saveOrUpdate与Update的作用 Hibernate Delete query Hibernate Basics ...

  5. Hibernate中对象的三种状态即save&lpar;&rpar;&comma;update&lpar;&rpar;&comma;saveOrUpdate&lpar;&rpar;的使用【转】

    当new一个user的时候,是处于瞬时状态 当调用session.save(user)的时候,user对象会放到session里,此时处于持久状态 当提交事务的时候,user对象才会存到DB中 当调用 ...

  6. save&lpar;&rpar;、saveOrUpdate&lpar;&rpar;、merge&lpar;&rpar;的区别

    一.Save() save()方法能够保存实体到数据库.假如两个实体之间有关系(例如employee表和address表有一对一关系),如果在没有事务的情况下调用这个方法保存employee这个实体, ...

  7. hibernate中save&comma;update&comma;saveorupdate

    save在添加用的时候  不会出现索引机制(即遍历目录  效率最高)update在修改时候要遍历  不存在则会异常saveorupdate是优先遍历  如果不存在则创建(效率最低)

  8. hibernate save update merge 区别

    1.save save的对象是临时对象,首先将对象写入缓存,使其变为持久对象. save函数底层使用的是Insert语句,因此如果数据库中已经存在相同ID的记录,那么会报错 2.update upda ...

  9. &lbrack;hibernate&rsqb;save&lpar;&rpar;与persist&lpar;&rpar;区别

    Hibernate 之所以提供与save()功能几乎完全类似的persist()方法,一方面是为了照顾JPA的用法习惯.另一方面,save()和 persist()方法还有一个区别:使用 save() ...

随机推荐

  1. oracle 自增长序列

    create or replace TRIGGER "METTINGUSER".TRG_PREPN BEFORE INSERT ON "PREPROJFUN" ...

  2. HTTP 错误 500&period;23 - Internal Server Error 解决方法

    分析原因:在安装完成后IIS已经支持ASP和ASP.NET 2.0,需要注意的是.NET站点的应用程序池应选用Classic .NET AppPool,而不能用默认的DefaultAppPool,否则 ...

  3. fibonacci数列的和取余(2)

    Maybe ACMers of HIT are always fond of fibonacci numbers, because it is so beautiful. Don't you thin ...

  4. 2014年辛星完全解读Javascript第二节

    本小节我们讲解一下Javascript的语法,虽然js语言非常简单,它的语法也相对好学一些,但是不学总之还是不会的,因此,我们来一探究竟把. ********注释************* 1.我们通 ...

  5. python&lowbar;8&lowbar;字典

    什么是字典? --key -value 的数据类型,找到key就可以找到对应的值 --字典形式:{key1:value1,key2:value2,......} #!/usr/bin/python3 ...

  6. elasticsearch 5&period;0以上不支持consistency 和 quorum

    从ES2.2升级到5.2后,原先执行put 带 consistency=all / quorum 参数的,都报错了,提示语法错误.. 百度查了一通,都没发现相关问题.无奈,还是查官方文档.. 发现这是 ...

  7. Bootstrap相关网站中简单的等待提醒

    一.在页面中加入如下代码 <div class="modal fade" tabindex="-1" role="dialog" id ...

  8. 创建自定义graphql-binding

    graphql-binding 是一个比较方便强大的工具,方便我们进行代码生成以及开发gateway的功能 项目初始化 使用prisma cli 使用脚手架 prisma init appdemo ? ...

  9. js获取上一页、当前页及域名url

    一个业务中可能会用到,跳转到另个页面后, 又后退回之前的页面,之前的页面上有个判断提示一定会出 网上搬了下代码 console.log("js获取当前域名"+window.loca ...

  10. vue项目使用vux框架配置教程

    吐槽下,这个vux配置教程那..写的实在是坑,也不搞个示例代码...想上天吗???? 正常安装的话...100%报错...以下是正确配置 1.项目里安装vux   npm install vux -- ...