Hibernate之底层原理

1.Hibernate简单介绍

Hibernate是一个开放源代码的对象关系映射框架,它对JDBC进行了非常轻量级的对象封装,它将POJO与数据库表建立映射关系,是一个全自动的orm框架,hibernate可以自动生成SQL语句,自动执行,使得Java程序员可以随心所欲的使用对象编程思维来操纵数据库。

2.Hibernate的优缺点

2.1 优点

  • 程序更加面向对象;
  • 提高了生产率;
  • 方便移植(修改配置文件);
  • 无侵入性。

2.2 缺点

  • 效率比JDBC略差;
  • 不适合批量操作。
  • 只能配置一种关联关系

2.3 无入侵性

简单理解,所谓侵入性,是指它没有侵入hibernate任何的API。完全采用普通的Java对象(POJO),而不必继承Hibernate的某个基类,或实现Hibernate的某个接口。

例如 : A是侵入性的,B代码中使用A,那么如果以后不用A了(用另外一个工具代替),必须修改B的代码。反之,如果A是非侵入性的,B不用A,用C了,代码不需要改,改改配置文件什么的,就可以了。

3.Hibernate有四种查询方案

3.1 根据ID查询

根据get,load方法,根据id查找对象。

例如:

load(Class theClass, Serializable id) 
load(Class theClass, Serializable id, LockMode lockMode)
load(Object object, Serializable id)  
  •  

3.2 HQL语句进行查询

HQL(hibernate query language),查询对象:Query。Query由Session里的createQuery()来产生一个查询。

例如:

 //不带参数的查询(这类比较简单)
 Query query=session.createQuery("select user from User as user");

 //第一种带参数的查询
 Query query=session.createQuery("select user from User as user where user.name=?");
 //假设name为传过来的参数
 query.setString(0,name)

 //第二种带参数的查询
 Query query=session.createQuery("select user from User as user where user.name=:name");
 query.setString("name",name)//假设name为传过来的参数(多个参数以此类推)  


利用Session接口的find查询,均返回list 
find(String query) 
find(String query, Object[] values, Type[] types) 
find(String query, Object value, Type type)      
//例如下面
List list=session.find("select user from Users as user where user.name=?",name,Hibernate.STRING)
List list=session.find("select user from Users as user where user.name=? and             user.pw=?",new Object[]{name,pw},new Type[]{Hibernate.STRING,Hibernate.STRING})
  •  

3.3 Criteria查询

Criteria(标准查询语言),查询对象:Criteria,查询条件:Criterion。

具体实例如下:省略 User 类

@Test  
public void test() {  
    // 获取Criteria实例对象  
    Criteria criteria = session.createCriteria(User.class);  

    // 查询出王姓员工且收入在3000到5000之间的  
    // 类似于HQL中 WHERE employeeName LIKE '王%' AND salary BETWEEN 3000 AND 5000  
    List emps = criteria.add(Restrictions.like("employeeName", "王%"))  
            .add(Restrictions.between("salary", 3000.0, 5000.0)).list();  

    // 查询出工资在4000以下或5000以上的王姓员工  
    // 可以通过Restrictions的or或and进行逻辑分组  
    emps = criteria.add(Restrictions.like("employeeName", "王%"))  
            .add(Restrictions.or(Restrictions.gt("salary", 5000D), Restrictions.lt("salary", 3000D))).list();  

    // 查询出岗位是软件工程师或测试工程师,且学历是硕士、本科或大专的员工有哪些  
    emps = criteria.add(Restrictions.in("position", new String[] { "软件工程师", "测试工程师" }))  
            .add(Restrictions.disjunction().add(Restrictions.eq("degree", "硕士")).add(Restrictions.eq("degree", "本科"))  
            .add(Restrictions.eq("degree", "大专")))  
            .list();  
}  
  •  

更多的细节,请看下面网络上的大神文章,这里不继续扩展。

3.4 通过SQL来查

查询对象:SQLQuery

Session.createSQLQuery();
  •  

最基本的SQL查询就是获得一个标量(数值)的列表

Session.createSQLQuery("SELECT * FROM CATS").list();
Session.createSQLQuery("SELECT ID, NAME, BIRTHDATE FROM CATS").list();
  •  

下面展示如何通过 addEntity() 让原生查询返回实体对象。

Session.createSQLQuery("SELECT * FROM CATS").addEntity(Cat.class);
Session.createSQLQuery("SELECT ID, NAME, BIRTHDATE FROM CATS").addEntity(Cat.class);
  •  

更多的细节,请看下面网络上的大神文章,这里不继续扩展。

4.Hibernate的工作原理和初始化

4.1工作原理

  • 配置好hibernate的配置文件和与类对应的配置文件后,启动服务器
  • 服务器通过实例化Configuration对象,读取hibernate.cfg.xml文件的配置内容,并根据相关的需求建好表或者和表建立好映射关系
  • 通过实例化的Configuration对象就可以建立sessionFactory实例,进一步,通过sessionFactory实例可以创建 session对象
  • 得到session之后,便可以对数据库进行增删改查操作了,除了比较复杂的全文搜索外,简单的操作都可以通过hibernate封装好的 session内置方法来实现
  • 此外,还可以通过事物管理,表的关联来实现较为复杂的数据库设计
    优点:hibernate相当于java类和数据库表之间沟通的桥梁,通过这座桥我们就可以做很多事情了

4.2 初始化

1.通过Configuration config = new Configuration().configure();//读取并解析hibernate.cfg.xml配置文件
2.由hibernate.cfg.xml中的读取并解析映射信息
3.通过SessionFactory sf = config.buildSessionFactory();//创建SessionFactory
4.Session session = sf.openSession();//打开Sesssion
5.Transaction tx = session.beginTransaction();//创建并启动事务Transation
6.persistent operate操作数据,持久化操作
7.tx.commit();//提交事务
8.关闭Session
9.关闭SesstionFactory

5.Hibernate的3种对象状态

5.1 临时状态(或者叫瞬时状态Transient)

由new命令开辟内存空间的java对象,例如:

User user=new User();
  •  

临时对象在内存孤立存在,它是携带信息的载体,不和数据库的数据有任何关联关系。

5.2 持久状态(Persistent)

该状态的对象在数据库中具有对应的记录,并拥有一个持久化标识.通过session的 get()、load() 等方法获得的对象都是持久对象。持久化对象被修改变更后,不会马上同步到数据库,直到数据库事务提交。在同步之前,持久化对象是脏的。

5.3 游离状态(或者叫脱管状态Detached)

当与某持久对象关联的 session 被关闭后,该持久对象转变为游离对象.当游离对象被重新关联到session上 时,又再次转变成持久对象(在Detached其间的改动将被持久化到数据库中)。游离对象拥有数据库的识别值,但已不在持久化管理范围之内。

6.Hibernate对象状态转换

6.1 转换图片

对象状态

6.2 对象状态转换说明

临时状态或者瞬时状态 (transient)
- 不处于Session 缓存中,存在内存中
- 数据库中没有对象记录

Java如何进入临时状态
- 通过new语句刚创建一个对象时
- 当调用Session 的delete()方法,从 Session 缓存中删除一个对象时。

持久化状态(persisted)
- 处于Session 缓存中。
- 持久化对象数据库中有对象记录。
- Session 在特定时刻会保持二者同步。

Java如何进入持久化状态
- Session 的save()把临时 —>>> 持久化状态。
- Session 的load(),get()方法返回的对象。
- Session 的find()返回的list集合中存放的对象。
- Session 的update(),saveOrupdate()使游离 —>>> 持久化。

游离状态或者托管状态(detached)
- 不再位于 Session 缓存中.
- 游离对象由持久化状态转变而来,数据库中可能还有对应记录。

Java如何进入持久化状态 —>>> 游离状态
- Session 的close()方法
- Session 的evict()方法,从缓存中删除一个对象,提高性能,少用。

这里写图片描述

7.Hibernate缓存

7.1为什么需要缓存

Hibernate是一个持久层框架,经常访问物理数据库。为了降低应用程序对物理数据源访问的频次,从而提高应用程序的运行性能。缓存内的数据是对物理数据源中的数据的复制,应用程序在运行时从缓存读写数据,在特定的时刻或事件会同步缓存和物理数据源的数据。

7.2 Hibernate缓存分类

Hibernate一级缓存又称为 Session 的缓存:

  • Session内置不能被卸载
  • Session的缓存是事务范围的缓存
  • Session对象的生命周期通常对应一个数据库事务或者一个应用事务

Hibernate二级缓存又称为“SessionFactory的缓存:

第二级缓存是可选的,是一个可配置的插件,默认下SessionFactory不会启用这个插件。

7.2 Hibernate缓存查询机制

  • 当Hibernate根据ID访问数据对象的时候,首先从Session一级缓存中查;
  • 查不到,如果配置了二级缓存,那么从二级缓存中查;
  • 如果都查不到,再查询数据库,把结果按照ID放入到缓存
  • 删除、更新、增加数据的时候,同时更新缓存。

猜你喜欢

转载自blog.csdn.net/qq_41344503/article/details/81127104