`
306781704
  • 浏览: 87222 次
  • 性别: Icon_minigender_1
  • 来自: 长沙
社区版块
存档分类
最新评论

hibernate几个方法的解释

阅读更多

(2)session.merge ()方法

该方法将修改表中记录,其所需要的实体状态为脱管状态,但是注意,它并不影响调用方法前后的状态,也即该实体依然是脱管状,见例6.4。

例6.4:session.merge ()方法对状态的变化

     public void run() {

          //创建UserInfo实例

          UserInfo userInfo = new UserInfo();

          //使之成为脱管状态

          userInfo.setId(11112);

          userInfo.setName("RW3");

          userInfo.setSex("M");

          //创建UserInfo实例

          UserInfo userInfo2 = new UserInfo();

          //使之成为脱管状态

          userInfo2.setId(11112);

          userInfo2.setName("RW4");

          userInfo2.setSex("F");

          //启动Session

          Session session = HibernateSessionFactory.currentSession();

          //启动事务

          Transaction tx = session.beginTransaction();

          //调用merge方法,此时UserInfo实体状态并没有被持久化

          session.merge(userInfo);

          //调用merge方法,此时UserInfo实体状态并没有被持久化

          //但是数据库中的记录被更新了

          ①session.merge(userInfo2);

          //merge方法与update方法的差别在于针对同样的操作update方法会报错

          //原因在于update方法使得实体状态成为了持久化状态,而Session中不允许两个持久化实体有同样的持久化标识

          ②//session.update(userInfo);

          //session.update(userInfo2);

         //以下两句不会发送SQL,因为userInfo2不是持久化状态的实体

         ③userInfo2.setName("RW5");

          userInfo2.setSex("M");

          //提交事务

          tx.commit();

          //关闭Hibernate Session

          HibernateSessionFactory.closeSession();

     }

针对该段代码将执行如下SQL语句:

Hibernate:

/* ①session.merge(userInfo2)的动作 */

select

        userinfo0_.id as id0_0_,

        userinfo0_.NAME as NAME0_0_,

        userinfo0_.SEX as SEX0_0_,

        userinfo0_.roomid as roomid0_0_

    from

        userinfo userinfo0_

    where

        userinfo0_.id=?

Hibernate:

/* ①session.merge(userInfo2)的动作 */

update

            userinfo

        set

            NAME=?,

            SEX=?,

            roomid=?

        where

            id=?

session.merge()方法会首先发送一句select语句,去数据库端获取UserInfo持久化标识所对应的表记录;然后自动生成一个持久化状态的UserInfo实体,与脱管状态的UserInfo实体做比较是否有所改变;一旦发生了改变,才会发送update语句执行更新。而按执行顺序,若两句session.merge()方法针对同一个脱管状态的UserInfo实体,那其结果只会执行最后一个session.merge()方法所发出的update语句。即使执行了session.merge()方法,UserInfo实体依然是脱管状态,因此③userInfo2. setName("RW5")的语句不会同步数据库中的表。

(3)session.lock()方法

他为了解决事务处理而使用,它会将实体从脱管状态转变为持久化状态。但是值得注意的是,调用session.lock()方法后,脱管状态的实体信息不会同步到数据库,而是会从数据库中返回该持久化状态。即使在脱管状态对实体属性进行了修改,一旦调用了session.lock()方法,这种修改就成了无效,见例6.5。

例6.5:session.lock()方法对状态的变化

     public void run() {

          //创建UserInfo实例

          UserInfo userInfo = new UserInfo();

          //使之成为脱管状态

          userInfo.setId(11112);

          userInfo.setName("RW3");

          userInfo.setSex("M");

          //启动Session

          Session session = HibernateSessionFactory.currentSession();

          //启动事务

          Transaction tx = session.beginTransaction();

         //发送select获取数据库中的当前记录(执行的SQL根据LockMode不同有不同的方式)

         //UserInfo实体将从脱管状态转变为持久化状态

          ①session.lock(userInfo,LockMode.UPGRADE_NOWAIT);

          //对当前UserInfo实体进行更新将同步数据库中的记录

          ②userInfo.setName("RW8");

          //提交事务

          tx.commit();

          //关闭Hibernate Session

          HibernateSessionFactory.closeSession();

     }

针对该段代码将执行如下SQL语句:

Hibernate:

/* ①session.lock(userInfo,LockMode.UPGRADE_NOWAIT)的动作 */

select

        id

    from

        userinfo

    where

        id =? for update

            nowait

Hibernate:

/*②userInfo.setName("RW8")的动作 */

update

            userinfo

        set

            NAME=?,

            SEX=?,

            roomid=?

        where

            id=?

session.lock()方法并不是为了将脱管状态的对象转变为持久化状态,而是为了事务处理。

(4)session.saveOrUpdate()方法

它是Hibernate提供的既可以新增也可以更新的方法,该方法使实体状态从脱管或瞬时直接变成持久化。session.saveOrUpdate()方法对实体的持久化标识非常敏感。当实体持久化标识存在,就会发送update SQL,当持久化标识不存在,就会发送insert SQL,见例6.6。

例6.6:session.saveOrUpdate()方法对状态的变化

     public void run() {

          //创建UserInfo实例

          UserInfo userInfo = new UserInfo();

          //使之成为脱管状态

          userInfo.setId(11112);

          userInfo.setName("RW3");

          userInfo.setSex("M");

          //创建UserInfo实例,其为瞬时状态

          UserInfo userInfo2 = new UserInfo();

          userInfo2.setName("RW3");

          userInfo2.setSex("M");

          //启动Session

          Session session = HibernateSessionFactory.currentSession();

          //启动事务

          Transaction tx = session.beginTransaction();

          //UserInfo存在持久化标识,因此为新增,从瞬时状态成为持久化状态

          ①session.saveOrUpdate(userInfo2);

          //同步数据库表记录

          ②userInfo2.setName("RW9");

          //UserInfo存在持久化标识,因此为修改,从脱管状态成为持久化状态

          ③session.saveOrUpdate(userInfo);

          //同步数据库表记录

          ④userInfo.setName("RW10");

          //提交事务

          tx.commit();

          //关闭Hibernate Session

          HibernateSessionFactory.closeSession();

     }

针对该段代码将执行如下SQL语句:

Hibernate:

/* ①session.saveOrUpdate(userInfo2)的动作 */

insert

        into

            userinfo

            (NAME, SEX, roomid, id)

        values

            (?, ?, ?, ?)

Hibernate:

/* ②session.saveOrUpdate(userInfo)的动作 */

update

            userinfo

        set

            NAME=?,

            SEX=?,

            roomid=?

        where

            id=?

Hibernate:

/* ③session.saveOrUpdate(userInfo)的动作 */

update

            userinfo

        set

            NAME=?,

            SEX=?,

            roomid=?

        where

            id=?

根据代码的执行,对同一持久化UserInfo属性需要改变多次,那只会以最后的属性为准,因此③session.saveOrUpdate(userInfo)和④userInfo.setName("RW10")虽然从理论上需要发送两句update SQL到数据库,但其实只会产生一句。

(5)session.createQuery()方法

它为HQL语句调用,HQL(HibernateQusery Language)是Hibernate框架自定义的一种面向对象的语言,类似SQL语言,用以与数据库进行交互。Hibernate将HQL解析成SQL语句与数据库交互。HQL被执行后,其所关系到的实体对象将从瞬时状态转变为脱管状态,见例6.7。

例6.7:session.createQuery()方法对状态的变化

//一个内部类,作为SQL查询的参数传递

     class RoomDTO {

          Long id;

          public Long getId() {

               return id;

          }

          public void setId(Long id) {

               this.id = id;

          }

     }

    

     public void run() {

          // 创建一个JavaBean作为参数传递

          RoomDTO roomDTO = new RoomDTO();

          //设置id属性的值

          roomDTO.setId(1L);

          // 启动Session

          Session session = HibernateSessionFactory.currentSession();

          // 启动事务

          Transaction tx = session.beginTransaction();

          //session.createQuery方法作为HQL查询的执行

          //其中setProperties方法作为":id"的参数传递,要求roomDTO实例中必须包含id

          //属性和getId、setId方法

          //由于SQL中包含有3个实体:room、room.id、userinfo,因此返回的结果将是对象数组

          ①Iterator i = session

                    .createQuery(

                              "select room, room.id, userinfo from Room room, UserInfo userinfo where room.id = userinfo.room.id and room.id = :id")

                    .setProperties(roomDTO).iterate();

          //通过迭代将3个实体对象转型,得到最终结果

          //其中Room实体和UserInfo实体对应的实体状态为脱管,roomid则为一个Long类型

          while (i.hasNext()) {

               //获取对象数组转型

               Object[] object = (Object[]) i.next();

               //获取脱管状态的Room实体

               ②Room roomr = (Room) object[0];

               System.out.println(roomr.getName());

               System.out.println(roomr.getRoomnumber());

               //获取roomid

               ③Long roomid = (Long) object[1];

               System.out.println(roomid);

               //获取脱管状态的UserInfo实体

               ④UserInfo userinfor = (UserInfo) object[2];

               System.out.println(userinfor.getName());

               System.out.println(userinfor.getSex());

          }

          // 提交事务

          tx.commit();

          // 关闭Hibernate Session

          HibernateSessionFactory.closeSession();

     }

针对该段代码将执行如下SQL语句:

Hibernate:

/* 执行

①select

        room,

        room.id,

        userinfo

    from

        Room room,

        UserInfo userinfo

    where

        room.id = userinfo.room.id

        and room.id = :id

的HQL语句 */

select

            room0_.id as col_0_0_,

            room0_.id as col_1_0_,

            userinfo1_.id as col_2_0_

        from

            room room0_,

            userinfo userinfo1_

        where

            room0_.id=userinfo1_.roomid

            and room0_.id=?

Hibernate:

/* ②Room roomr = (Room) object[0]的动作(数据库中有一条记录,取第一条) */

select

        room0_.id as id1_0_,

        room0_.NAME as NAME1_0_,

        room0_.roomnumber as roomnumber1_0_

    from

        room room0_

    where

        room0_.id=?

Hibernate:

/* ④UserInfo userinfor = (UserInfo) object[2]的动作(数据库中有两条记录,取第一条) */

select

        userinfo0_.id as id0_0_,

        userinfo0_.NAME as NAME0_0_,

        userinfo0_.SEX as SEX0_0_,

        userinfo0_.roomid as roomid0_0_

    from

        userinfo userinfo0_

    where

        userinfo0_.id=?

Hibernate:

/* ④UserInfo userinfor=(UserInfo)object[2]的动作(数据库中有两条记录,取第二条)*/

select

        userinfo0_.id as id0_0_,

        userinfo0_.NAME as NAME0_0_,

        userinfo0_.SEX as SEX0_0_,

        userinfo0_.roomid as roomid0_0_

    from

        userinfo userinfo0_

    where

        userinfo0_.id=?

可以看到,Hibernate在执行这段代码的HQL时,并不会一次性把所有的room表和userinfo表的字段都捞取出来,而是先获取其主键。在之后真正要使用这两个表所对应的实体对象(Room和UserInfo)时,才会调用select语句去获取其所有字段,这是“延时求值”的机制在起作用。session.createQuery()方法不会使实体成为持久化状态,因此对Room和UserInfo的实体属性进行改变不会同步数据库。

调用createQuery()方法执行HQL时,有多种方式可以传递参数,本例提供了一种常见的方式——setProperties()。


说明 除了createQuery()方法外,Hibernate还提供了外置命名查询(getNameQuery()方法)、结果集过滤(createFilter()方法)、条件查询(createCriteria()方法)、原生SQL查询(createSQLQuery()方法)来实现抓取数据。

6.1.6 结语
在本小节中,将对Hibernate中的持久化方法做一个总结,也作为本问题的结语。

(1)瞬时—脱管状态的方法有以下几种。

· 直接将实体的持久化标识进行改变。

· 调用session.createQuery()方法。

· 调用session.getNameQuery()方法。

· 调用session.createFilter()方法。

· 调用session.createCriteria()方法。

· 调用session.createSQLQuery()方法。

(2)瞬时—持久化状态的方法有以下几种。

· 调用session.save()方法。

· 调用session.saveOrUpdate()方法。

(3)脱管—持久化状态的方法有以下几种。

· 调用session.load()方法。

· 调用session.lock()方法。

· 调用session.update()方法。

· 调用session.saveOrUpdate()方法。

(4)脱管—瞬时状态的方法有以下几种。

· 直接将实体的持久化标识清除。

· 调用session.delete()方法。

(5)持久化—脱管状态的方法:关闭Hibernate Session。

(6)持久化—瞬时状态的方法。调用session.delete()方法。

(7)脱管状态-脱管状态但影响数据库记录的方法:调用session.merger()方法。


摘自:http://book.csdn.net/bookfiles/563/10056318681.shtml

分享到:
评论

相关推荐

    hibernate注解中的名词解释

    以下是对几个关键注解的详细解释: 1. **@Entity**: 此注解用于标记一个Java类为实体Bean,这意味着这个类的实例将会被持久化到数据库中。实体Bean通常代表数据库中的一个表,而类的属性则对应表中的列。 2. **@Id...

    Hibernate几个常见错误解决方式

    解决方法是检查Hibernate的映射文件(如.hbm.xml或使用注解的方式),确保实体类中的`isupdate`属性正确地映射到了数据库表的相应列。如果数据库中确实没有这个字段,需要更新实体类或者删除不必要的映射。 2. **...

    Hibernate方法的实现

    在使用Hibernate前,我们需要配置Hibernate的环境,这通常涉及到以下几个步骤: - 引入Hibernate的jar包。 - 创建hibernate.cfg.xml配置文件,设置数据库连接信息(如URL、用户名、密码等)以及持久化类信息。 -...

    Hibernate操作数据库的方法

    Hibernate操作数据库时,主要通过以下几个核心组件来实现持久层操作: 1. **Session**:Session是Hibernate用来和数据库进行交互的一个轻量级对象,它相当于JDBC中的Connection。Session对象是应用程序与数据库进行...

    Hibernate简单例子几个

    在 "Hibernate简单例子几个" 的描述中提到了 "many-to-one",这是 Hibernate 中的一种关联映射类型,表示多对一的关系。比如,一个部门可以有多名员工,而每个员工只属于一个部门。在实体类中,我们可以使用 `@...

    Hibernate映射导致的几个异常

    避免这个问题的方法是在访问懒加载属性之前确保Session仍然打开,或者显式地在需要时加载属性,如使用`Hibernate.initialize()`方法。 ### 5. MappingException - **MappingException: Error reading resource**:...

    Hibernate的Template方法以及SQL简介

    `HibernateTemplate`提供了多种用于操作持久化对象的方法,这些方法可以分为几大类:数据插入、数据更新、数据删除、数据查询等。接下来将详细介绍其中的几种核心方法: 1. **删除操作** - `void delete(Object ...

    Hibernate实践例子程序

    一个最简单的Hibernate project(不涉及Struts, Tomcat, XDoclet,JBoss等东东)必须的几个东东: 1. Hibernate工具包。 2. JDBC数据库连接驱动。以mysql为例,mysql-connector-java-3.1.×-bin.jar。 3. 配置...

    Hibernate连表查询 Hibernate连表查询

    通过对HBM映射文件的理解、DAO层的连表查询实现以及查询结果的处理等几个方面进行了详细解释。在实际项目中,合理运用这些技巧可以极大地提高开发效率和代码质量。需要注意的是,在编写连表查询时要充分考虑性能问题...

    Hibernate几种获数据方法比较

    本文将深入探讨Hibernate中的几种主要的数据获取方式,包括`Session.get()`与`Session.load()`、`Query.iterator()`与`Query.list()`以及使用HQL(Hibernate Query Language)进行查询。 ### 1. `Session.get()`与`...

    Hibernate常用的jar包和两个配置文件

    在Java Web开发中,Hibernate是一个非常重要的对象关系映射(ORM)框架,它极大地简化了数据库操作,使得开发者可以使用面向对象的方式来处理数据库事务。本篇文章将深入探讨Hibernate框架中常用的jar包以及两个核心...

    Hibernate查询解决方案

    本文将详细介绍 Hibernate 提供的几种查询方式及其应用场景。 #### 二、重要知识点 ##### 1. Session 对象与查询操作 在 Hibernate 中,所有持久化操作都需要通过 Session 对象来完成,包括查询操作。Session ...

    Hibernate入门案例源码

    在MyEclipse和Eclipse中,配置Hibernate通常包括以下几个步骤: 1. 添加Hibernate库:你需要导入Hibernate的jar文件或者通过Maven/Gradle等构建工具管理依赖。 2. 配置Hibernate:创建`hibernate.cfg.xml`配置文件,...

    优化Hibernate性能的几点建议

    ### 优化Hibernate性能的几点建议 ...总之,通过上述几个方面的优化措施,可以显著提高基于Hibernate构建的应用程序的性能。在实际应用中,还需要结合具体的业务场景和技术栈来灵活调整这些策略。

    spring与hibernate几个外载常用包

    在给定的标题“spring与hibernate几个外载常用包”中,我们可以推断出这些文件是与这两个框架的集成和日志管理相关的。接下来,我们将详细探讨每个文件的作用及其在Spring和Hibernate生态系统中的角色。 首先,`slf...

    hibernate

    #### 二、几个重要的类 **1. Configuration 类** - **作用**:负责管理 Hibernate 的配置信息。当创建 `SessionFactory` 时,`Configuration` 实例可以指定使用哪个映射文档。 - **用法**:应用程序通常只创建一个...

    hibernate中文参考文档

    7. 组件映射:当一个类的属性是由其他几个类的属性组合而成时,可以使用组件映射。这部分内容将说明如何实现组件映射,以组合不同的属性到一个复合对象。 8. 继承映射:Hibernate支持不同类型的继承映射策略,包括...

    Hibernate几种连接池的比较

    在选择连接池时,通常会考虑以下几个因素: - 性能:C3P0和Proxool都比Hibernate内置的连接池表现更好。 - 稳定性:C3P0和DBCP经过了广泛测试,通常被认为是稳定的。 - 功能和灵活性:C3P0和Proxool提供了丰富的配置...

    Hibernate完整使用教程

    Hibernate 的原理可以概括为以下几个步骤: * 对象模型的定义 * 对象关系映射的定义 * SQL 语句的生成 * 数据库交互的处理 Hibernate 创始人 Hibernate 的创始人是 Gavin King,他是一名澳大利亚的软件开发者。...

    Hibernate使用指南

    使用 Hibernate 的步骤主要包括以下几个步骤: * 导入 jar 包:在项目中添加 Hibernate 的 jar 包,以便使用 Hibernate。 * 添加配置文件:在项目的 src 目录下添加 hibernate.cfg.xml 配置文件,以便配置 ...

Global site tag (gtag.js) - Google Analytics