开心一笑

【老婆想减肥,让老公帮她买减肥药,老公:吃药伤身,现在挺好,多有肉感啊。儿子:其实真有点胖,老公立刻瞪了儿子一眼:小孩子不知道别乱讲,你妈妈这身材我喜欢。然后,趁老婆不注意狠狠的教训了儿子:以后说话注意点,减肥药很贵的,你要告诉你妈,少吃点饭不就减下去了。都是套路啊……】

提出问题

关于hibernate,个人的7点简单整理和总结???


Hibernate之底层原理的7点整理和总结_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 转换图片


Hibernate之底层原理的7点整理和总结_缓存_02

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()方法,从缓存中删除一个对象,提高性能,少用。


Hibernate之底层原理的7点整理和总结_缓存_03

7.Hibernate缓存

7.1为什么需要缓存

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

7.2 Hibernate缓存分类

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

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

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

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

7.2 Hibernate缓存查询机制

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

读书感悟

来自村上春树《海边的卡夫卡》

  • 从沙尘暴中逃出的你已不再是跨入沙尘暴时的你。
  • “希望你记住我”,佐伯说,“只要有你记住我,被其他所有人忘掉 都无所谓。”
  • 不能用语言准确表达的东西,最好完全不说。
  • 人不是因其缺点,而是因其优点被拖入更大的悲剧之中的。
  • 大凡事物必有顺序,看的太超前了不行。看的太超前,势必忽视脚下,人往往跌倒。可另一方面,光看脚下也不行。不看好前面,会撞上什么。所以么,要在多少往前看的同时按部就班处理眼下事物。这点至为关键,无论做什么。
  • 我们大家都在持续失去种种宝贵的东西,宝贵的机会和可能性,无法挽回的感情,这是生存的一个意义。
  • 如果拥有令人吃惊的了不起的想法的是你一个人,那么在深重的黑暗中往来彷徨的也必定是你一个人。
  • 问乃一时之耻,不问乃一生之耻。

经典故事

【一家酒店经营得很好,人气旺盛、财源广进。酒店的老总准备开展另外一项业务,由于没有太多的精力管理这家酒店,打算在现有的三个部门经理中物色一位总经理。 
老总问第一位部门经理:“是先有鸡还是先有蛋?” 
第一位部门经理不加思索地答道:“先有鸡。” 
老总接着问第二位部门经理:“是先有鸡还是先有蛋?” 
第二位部门经理胸有成竹地答道:“先有蛋。” 
这时,老总向最后一位部门经理说道:“你来说说,是先有鸡还是先有蛋?” 
第三位部门经理认真地答道:“客人先点鸡,就先有鸡;客人先点蛋,就先有蛋。” 
老总笑了。他决定将第三位部门经理升任为这家酒店的总经理。 
【心语】就事论事,往往容易局限在一个小圈子里,这就是常说的:惯性思维“。跳不出来时,就找不到得理事情的正确方法;相反,当我们换个角度跳出原来的惯性思维的框框时,我们就走上了一条新路,即:柳暗花明又一春。】


其他

如果有带给你一丝丝小快乐,就让快乐继续传递下去,欢迎点赞、顶、欢迎留下宝贵的意见、多谢支持!

阿里云国内75折 回扣 微信号:monov8
阿里云国际,腾讯云国际,低至75折。AWS 93折 免费开户实名账号 代冲值 优惠多多 微信号:monov8 飞机:@monov6