hibernate

hibernate

 

一、hibernate实体的注意事项:

1、持久化类需要提供无参数的构造方法

2、持久化类的属性(set/get) 需要与数据库表的列对应

3、持久化类的属性要尽量使用包装类的类型(包装类默认初始化为null而不是)

4、持久化类要有一个唯一标识OID 与数据库表主键对应

5、持久化类坚决不能用final修饰(后面要被继承的)

 

二、主键生成策略

increment:hibernate负责生成主键。保存前先查询表中ID的最大值,然后加一作为新的主键,会有线程安全的问题,只在测试时使用

identity:  由数据库创建主键(主键自增)。

sequence:  由数据库创建主键,序列,oracle中的对象。

hilo:      hibernate负责生成主键,高低位算法。

native:    自动三选一。identity|sequence|hilo。最后一个基本不用

uuid:     hibernate负责生成主键,使用uuid生成随机字符串作为主键,注意:uuid必须为字符串类型的主键

assigned:  由开发人员手动指定ID值,实体使用自然主键时,使用该策略

自然主键|代理主键

自然主键:实体中,本身就具备业务属性ID的特点(非空不能重复),该属性可以直接用作主键

代理主键: 不具备符合ID特点的业务属性,所以为其加上一列,不具备业务意义

 

三、对象的三种状态

瞬时状态                  没有ID值(null),没有与session关联的

持久化状态               有ID值,与session有关联

托管/游离状态         有ID值,没有与session关联

save:将瞬时状态转化成持久化状态

update:将游离状态转换成持久化状态

持久化状态会自动与数据库同步,将需要持久化到数据库的对象,只需要转换成持久化状态即可

close 将持久化状态转换成游离状态

delete 将持久化状态转成瞬时状态

一级缓存,(缓存和快照)

查询时体现--查询同一个ID只发送一条语句

修改时体现---如果改的值与数据库本来的值相同,什么也不做;如果改的值与数据库本来的值不同,则修改

ACID:原子性,一致性,隔离性,持久性

不同的session的缓存区不是同一个,数据不能共享

调用session.clear() 方法,可以强制清除Session缓存

1)Hibenate中一级缓存,也叫做session的缓存,它可以在session范围内减少数据库的访问次数! 只在session范围有效! Session关闭,一级缓存失效!

2)当调用session的save/saveOrUpdate/get/load/list/iterator方法的时候,都会把对象放入session的缓存中

3)Session的缓存由hibernate维护, 用户不能操作缓存内容; 如果想操作缓存内容,必须通过hibernate提供的evit/clear方法操作。

4)只在(当前)session范围有效,作用时间短,效果不是特别明显!

5)适合在短时间内多次操作数据库,效果比较明显!

6)session.flush(); 让一级缓存与数据库同步,一条update语句

7)session.evict(arg0); 清空一级缓存中指定的对象

8)session.clear(); 清空一级缓存中缓存的所有对象 

9)list() 默认情况只会放入缓存,不会从一级缓存中取,配置查询缓存,可以让list()查询从二级缓存中取数据。

在什么情况用上面方法?

批量操作使用使用:

Session.flush(); 先与数据库同步

Session.clear(); 再清空一级缓存内容

 

 

四、hibernate事务管理

事务并发问题:

脏读:读取到了其他事务修改但是还未提交的数据

不可重复读:在两次连续读取期间,数据被其他事务修改并提交,造成两次连续读取结果不一致。

幻读:删除整表记录时,有人插入了一条记录,导致删除后还剩余一条记录

事务的隔离级别:

1、读未提交    ----可以读到未提交的数据:123

2、读已提交   -----只能读到已提交的数据:23    (oracle默认)

4、可重复读     ----3    (mysql默认)

8、串行化         ----不允许并发

 

事务隔离级别修改

在配置文件中修改

使用hibernate在项目中管理事务

getCurrentSession需要配置,不然会出错

<!-- 配置getCurrentSession方法中,session与线程绑定 -->

<property name="hibernate.current_session_context_class" >thread</property>

该方法返回的session会在事务提交后自动关闭,无需手动关闭,如果再手动关闭,会抛出异常

五、session的批量条件查询

HOL: Hibernate Query Language=>hibernate查询语言,是hibernate框架独有的查询语言,面向对象的查询语言

Criteria:面向对象的无语句查询,调用方法,hibernate独有   (QBC查询)

原生SQL:

 

****mysql中limit默认会按照主键排序然后取值

 

 

六、常用HQL语句:

    // HQL: Hibernate Query Language.  

    // 特点:  

    // >> 1,与SQL相似,SQL中的语法基本上都可以直接使用。  

    // >> 2,SQL查询的是表和表中的列;HQL查询的是对象与对象中的属性。  

    // >> 3,HQL的关键字不区分大小写,类名与属性名是区分大小写的。  

    // >> 4,SELECT可以省略.         

              

    // 1,简单的查询,Employee为实体名而不是数据库中的表名(面向对象特性)  

    hql = "FROM Employee";  

    hql = "FROM Employee AS e"; // 使用别名  

    hql = "FROM Employee e"; // 使用别名,as关键字可省略  

      

    // 2,带上过滤条件的(可以使用别名):Where  

    hql = "FROM Employee WHERE id<10";  

    hql = "FROM Employee e WHERE e.id<10";  

    hql = "FROM Employee e WHERE e.id<10 AND e.id>5";  

      

    // 3,带上排序条件的:Order By  

    hql = "FROM Employee e WHERE e.id<10 ORDER BY e.name";  

    hql = "FROM Employee e WHERE e.id<10 ORDER BY e.name DESC";  

    hql = "FROM Employee e WHERE e.id<10 ORDER BY e.name DESC, id ASC";  

      

    // 4,指定select子句(不可以使用select *)  

    hql = "SELECT e FROM Employee e"; // 相当于"FROM Employee e"  

    hql = "SELECT e.name FROM Employee e"; // 只查询一个列,返回的集合的元素类型就是这个属性的类型  

    hql = "SELECT e.id,e.name FROM Employee e"; // 查询多个列,返回的集合的元素类型是Object数组  

    hql = "SELECT new Employee(e.id,e.name) FROM Employee e"; // 可以使用new语法,指定把查询出的部分属性封装到对象中  

      

    // 5,执行查询,获得结果(list、uniqueResult、分页 )  

    Query query = session.createQuery("FROM Employee e WHERE id<3");  

    query.setFirstResult(0);  

    query.setMaxResults(10); // 等同于 limit 0,10  

    //两种查询结果list、uniqueResult  

    // List list = query.list(); // 查询的结果是一个List集合  

    // Employee employee = (Employee) query.uniqueResult();// 查询的结果是唯一的一个结果,当结果有多个,就会抛异常  

      

    // 6,方法链  

    List list = session.createQuery(//  

            "FROM Employee e")//  

            .setFirstResult(0)//  

            .setMaxResults(10)//  

            .list();  

      

    // 7,聚集函数:count(), max(), min(), avg(), sum()  

    hql = "SELECT COUNT(*) FROM Employee"; // 返回的结果是Long型的  

    hql = "SELECT min(id) FROM Employee"; // 返回的结果是id属性的类型  

      

    //8,分组: Group By ... Having  

    hql = "SELECT e.name,COUNT(e.id) FROM Employee e GROUP BY e.name";  

    hql = "SELECT e.name,COUNT(e.id) FROM Employee e GROUP BY e.name HAVING count(e.id)>1";  

    hql = "SELECT e.name,COUNT(e.id) FROM Employee e WHERE id<9 GROUP BY e.name HAVING count(e.id)>1";  

    hql = "SELECT e.name,COUNT(e.id) " + //  

         "FROM Employee e " + //  

         "WHERE id<9 " + //  

         "GROUP BY e.name " + //  

         "HAVING count(e.id)>1 " + //  

         "ORDER BY count(e.id) ASC";  

    hql = "SELECT e.name,COUNT(e.id) AS c " + //  

         "FROM Employee e " + //  

         "WHERE id<9 " + //  

         "GROUP BY e.name " + //  

         "HAVING count(e.id)>1 " + // 在having子句中不能使用列别名  

         "ORDER BY c ASC"; // 在orderby子句中可以使用列别名  

      

    // 9,连接查询 / HQL是面向对象的查询  

     //>> 内连接(inner关键字可以省略)  

     hql = "SELECT e.id,e.name,d.name FROM Employee e JOIN e.department d";  

     hql = "SELECT e.id,e.name,d.name FROM Employee e INNER JOIN e.department d";  

     //>> 左外连接(outer关键字可以省略)  

     hql = "SELECT e.id,e.name,d.name FROM Employee e LEFT OUTER JOIN e.department d";  

     //>> 右外连接(outer关键字可以省略)  

     hql = "SELECT e.id,e.name,d.name FROM Employee e RIGHT JOIN e.department d";  

     //可以使用更方便的方法  

     hql = "SELECT e.id,e.name,e.department.name FROM Employee e";  

      

    // 10,查询时使用参数  

    // >> 方式一:使用'?'占位  

     hql = "FROM Employee e WHERE id BETWEEN ? AND ?";  

     List list2 = session.createQuery(hql)//  

         .setParameter(0, 5)// 设置参数,第1个参数的索引为0。  

         .setParameter(1, 15)//  

         .list();  

      

    // >> 方式二:使用变量名  

     hql = "FROM Employee e WHERE id BETWEEN :idMin AND :idMax";  

     List list3 = session.createQuery(hql)//  

         .setParameter("idMax", 15)//  

         .setParameter("idMin", 5)//  

         .list();  

      

    // 当参数是集合时,一定要使用setParameterList()设置参数值  

     hql = "FROM Employee e WHERE id IN (:ids)";  

     List list4 = session.createQuery(hql)//  

         .setParameterList("ids", new Object[] { 1, 2, 3, 5, 8, 100 })//  

         .list();  

      

    // 11,update与delete,不会通知Session缓存  

    // >> Update  

    int result = session.createQuery(//  

            "UPDATE Employee e SET e.name=? WHERE id>15")//  

            .setParameter(0, "无名氏")//  

            .executeUpdate(); // 返回int型的结果,表示影响了多少行。  

    // >> Delete  

    int result1 = session.createQuery(//  

            "DELETE FROM Employee e WHERE id>15")//  

            .executeUpdate(); // 返回int型的结果,表示影响了多少行。  

 

解耦通俗地说就是两个东西原来互相影响,现在让他们独立发展;核心思想还是最小职责,每个地方都只做一件事情;只要一个地方负责了多项事情,就存在解耦的可能。在系统每个层次都可以体现解耦的思想,比如在架构层面把存储和业务逻辑解耦,把动态页面和静态页面解耦;在模块层面把业务模块和统计模块解耦;在代码层面把多个功能解耦等等。解耦的思想很好,但是没必要为了解耦而解耦,还是要从业务需求以及系统定位出发,满足一段时间内系统发展的需要即可。

 

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值