Java EE五大框架之 Hibernate 3

1  Hibernate中的事务与并发

1.1  事务相关的概念

1. 什么是事务
* 事务就是逻辑上的一组操作,组成事务的各个执行单元,操作要么全都成功,要么全都失败.
* 转账的例子:冠希给美美转钱,扣钱,加钱。两个操作组成了一个事情!

2. 事务的特性
* 原子性 -- 事务不可分割.
* 一致性 -- 事务执行的前后数据的完整性保持一致.
* 隔离性 -- 一个事务执行的过程中,不应该受到其他的事务的干扰.
* 持久性 -- 事务一旦提交,数据就永久保持到数据库中.

3. 如果不考虑隔离性:引发一些读的问题
* 脏读 -- 一个事务读到了另一个事务未提交的数据.
* 不可重复读 -- 一个事务读到了另一个事务已经提交的update数据,导致多次查询结果不一致.
* 虚读 -- 一个事务读到了另一个事务已经提交的insert数据,导致多次查询结果不一致.

4. 通过设置数据库的隔离级别来解决上述读的问题
* 未提交读:以上的读的问题都有可能发生.
* 已提交读:避免脏读,但是不可重复读,虚读都有可能发生.
* 可重复读:避免脏读,不可重复读.但是虚读是有可能发生.
* 串行化:以上读的情况都可以避免.

5. 如果想在Hibernate的框架中来设置隔离级别,需要在hibernate.cfg.xml的配置文件中通过标签来配置
* 通过:hibernate.connection.isolation = 4 来配置
* 取值
* 1—Read uncommitted isolation
* 2—Read committed isolation
* 4—Repeatable read isolation

* 8—Serializable isolation


1.2  丢失更新的问题

    

1. 如果不考虑隔离性,也会产生写入数据的问题,这一类的问题叫丢失更新的问题。
2. 例如:两个事务同时对某一条记录做修改,就会引发丢失更新的问题。
* A事务和B事务同时获取到一条数据,同时再做修改
* 如果A事务修改完成后,提交了事务
* B事务修改完成后,不管是提交还是回滚,如果不做处理,都会对数据产生影响

3. 解决方案有两种
* 悲观锁
* 采用的是数据库提供的一种锁机制,如果采用做了这种机制,在SQL语句的后面添加 for update 子句
* 当A事务在操作该条记录时,会把该条记录锁起来,其他事务是不能操作这条记录的。
* 只有当A事务提交后,锁释放了,其他事务才能操作该条记录

* 乐观锁
* 采用版本号的机制来解决的。会给表结构添加一个字段version=0,默认值是0
* 当A事务在操作完该条记录,提交事务时,会先检查版本号,如果发生版本号的值相同时,才可以提交事务。同时会更新版本号version=1.
* 当B事务操作完该条记录时,提交事务时,会先检查版本号,如果发现版本不同时,程序会出现错误。

4. 使用Hibernate框架解决丢失更新的问题
* 悲观锁
* 使用session.get(Customer.class, 1,LockMode.UPGRADE); 方法

* 乐观锁
* 1.在对应的JavaBean中添加一个属性,名称可以是任意的。例如:private Integer version; 提供get和set方法

* 2.在映射的配置文件中,提供<version name="version"/>标签即可。


//解决“读”问题:设置事务隔离级别

//解决“写”问题:悲观锁 乐观锁


1.3  绑定本地的Session

1. 之前在讲JavaWEB的事务的时候,需要在业务层使用Connection来开启事务,
* 一种是通过参数的方式传递下去
* 另一种是把Connection绑定到ThreadLocal对象中


2. 现在的Hibernate框架中,使用session对象开启事务,所以需要来传递session对象,框架提供了ThreadLocal的方式
* 需要在hibernate.cfg.xml的配置文件中提供配置
* <property name="hibernate.current_session_context_class">thread</property>

* 重写HibernateUtil的工具类,使用SessionFactory的getCurrentSession()方法,获取当前的Session对象。并且该Session对象不用手动关闭,线程结束了,会自动关闭。
public static Session getCurrentSession(){
return factory.getCurrentSession();
}

* 注意:想使用getCurrentSession()方法,必须要先配置才能使用。

        这里有必要说明:一般不会在dao层使用transaction,事务被配置在service层上更为合理,因为业务层方法表示逻辑上的一个原子操作。因此,我们应该在Dao层获取sesssion,然后进行事务处理。



2  Hibernate框架的查询方式

2.1  Query的查询接口

Query执行HQL语句:

1.查询所有记录

    Query query = session.createQuery("from Customer");

    List<Customer> list = query.list();   

    for (Customer customer : list){

          System.out.println(customer);

    }


2.条件查询:

(1) 

   Query query = session.createQuery("from Customer where age > ?");

    query.setInteger(0, 18);

    List<Customer> list = query.list();

    for (Customer customer : list){

          System.out.println(customer);

    }

也可以指定名称:

   Query query = session.createQuery("from Customer where age > :aaa");

    query.setInteger("aaa", 18);

    List<Customer> list = query.list();

    for (Customer customer : list){

          System.out.println(customer);

    }


(2)

    Query query = session.createQuery("from Customer wherename = ?");

    query.setString(0, "小霏");

    List<Customer> list = query.list();

    for (Customer customer : list){

          System.out.println(customer);

    }

(3)

    Query query = session.createQuery("from Customer where name like ?");

    query.setString(0, "%小霞%");

    List<Customer> list = query.list();

    for (Customer customer : list){

          System.out.println(customer);

    }

    

3.条件查询:

    Query query = session.createQuery("from Customer wherename = :aaa and age = :bbb");

    query.setString("aaa", "李健");

    query.setInteger("bbb", 38);

    List<Customer> list = query.list();

    for (Customer customer : list){

          System.out.println(customer);

    }

   

4.分页查询:  //MySQL中的LIMIT

    Query query = session.createQuery("from Customer");

    query.setFirstResult(3);

    query.setMaxResults(3);

    List<Customer> list = query.list();

    for (Customer customer : list){

          System.out.println(customer);

    }


2.2  Criteria查询接口(做条件查询非常合适)

完全面向对象

Criteria:用来执行条件查询

1.查询所有记录

   //先获取到Criteria接口

    Criteria criteria = session.createCriteria(Customer.class);

    List<Customer> list = criteria.list();

    for (Customer customer : list){

          System.out.println(customer);

    }

   

2.条件查询

    Criteria criteria = session.createCriteria(Customer.class);

    //criteria是hibernate提供的条件查询的对象,想传入条件得使用工具类Restrictions,里边都是静态方法

    criteria.add(Restrictions.eq("name", "小霞"));

    //criteria.add(Restrictions.gt("age", 18));

    List<Customer> list = criteria.list();

    for (Customer customer : list){

          System.out.println(customer);

    }

   

 3.条件查询

    Criteria criteria = session.createCriteria(Customer.class);

    criteria.add(Restrictions.like("name", "%霞%"));

    criteria.add(Restrictions.gt("age", 18));

    List<Customer> list = criteria.list();

    for (Customer customer : list){

          System.out.println(customer);

    }

   

    // 4.分页查询

    Criteria criteria = session.createCriteria(Customer.class);

    criteria.setFirstResult(3);

    criteria.setMaxResults(3);

    List<Customer> list = criteria.list();

    for (Customer customer : list){

          System.out.println(customer);

    }


2.3  SQLQuery查询接口

1. 基本查询

        SQLQuery  sqlQuery = session.createSQLQuery("select * from cst_customer");

        List<Object[]>  list = sqlQuery.list();

        for(Object[] objects : list) {

            System.out.println(Arrays.toString(objects));

        }

       

        SQLQuery sqlQuery = session.createSQLQuery("select * from cst_customer");

        // 封装到对象中

        sqlQuery.addEntity(Customer.class);

        List<Customer> list = sqlQuery.list();

        for(Customer customer:list){

            System.out.println(customer);

        }


  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值