文档章节

Hibernate 查询

村长大神
 村长大神
发布于 2014/06/04 15:06
字数 561
阅读 104
收藏 3

本文主要通过源码,概述一下session的查询过程。

主要api有get、load。他们基本过程是类似的;

以get为例:

Session

public Object get(String entityName, Serializable id) throws HibernateException;

调用SessionImpl:

Java代码 复制代码 收藏代码
  1. public Object get(String entityName, Serializable id) throws HibernateException {
  2. LoadEvent event = new LoadEvent(id, entityName, false, this);
  3. boolean success = false;
  4. try {
  5. fireLoad(event, LoadEventListener.GET);
  6. success = true;
  7. return event.getResult();
  8. }
  9. finally {
  10. afterOperation(success);
  11. }
  12. }
public Object get(String entityName, Serializable id) throws HibernateException {
		LoadEvent event = new LoadEvent(id, entityName, false, this);
		boolean success = false;
		try {
			fireLoad(event, LoadEventListener.GET);
			success = true;
			return event.getResult();
		}
		finally {
			afterOperation(success);
		}
	}

fireLoad 主要是出发监听器;

主要获取过程就是 event.getResult();

event为LoadEvent,其实是一个很简单的类,其getResult也是非常简单,

Java代码 复制代码 收藏代码
  1. public Object getResult() {
  2. return result;
  3. }
  4. public void setResult(Object result) {
  5. this.result = result;
  6. }
public Object getResult() {
  return result;
 }
 public void setResult(Object result) {
  this.result = result;
 }

其中result其实是由 fireLoad(event, LoadEventListener.GET) 执行过程中赋值的

Java代码 复制代码 收藏代码
  1. private void fireLoad(LoadEvent event, LoadType loadType) {
  2. errorIfClosed();
  3. checkTransactionSynchStatus();
  4. LoadEventListener[] loadEventListener = listeners.getLoadEventListeners();
  5. for ( int i = 0; i < loadEventListener.length; i++ ) {
  6. loadEventListener[i].onLoad(event, loadType);
  7. }
  8. }
private void fireLoad(LoadEvent event, LoadType loadType) {
		errorIfClosed();
		checkTransactionSynchStatus();
		LoadEventListener[] loadEventListener = listeners.getLoadEventListeners();
		for ( int i = 0; i < loadEventListener.length; i++ ) {
			loadEventListener[i].onLoad(event, loadType);
		}
	}

当然,从这段代码中我们仍然看不出什么头绪,我们继续研究

loadEventListener[i].onLoad(event, loadType);

经查找发现loadEvent有一个listener :DefaultLoadEventListener

其onLoad方法主要调用

event.setResult( load(。。。。 ) ); 也就是 DefaultLoadEventListener 的load方法,而其中主要又调用

Object entity = doLoad(event, persister, keyToLoad, options); 来进行“查询”:

Java代码 复制代码 收藏代码
  1. protected Object doLoad(
  2. final LoadEvent event,
  3. final EntityPersister persister,
  4. final EntityKey keyToLoad,
  5. final LoadEventListener.LoadType options)
  6. throws HibernateException {
  7. //1
  8. Object entity = loadFromSessionCache(event, keyToLoad, options);
  9. if ( entity == REMOVED_ENTITY_MARKER ) {
  10. return null;
  11. }
  12. if ( entity != null ) {
  13. return entity;
  14. }
  15. //2
  16. entity = loadFromSecondLevelCache(event, persister, options);
  17. if ( entity != null ) {
  18. return entity;
  19. }
  20. //3
  21. return loadFromDatasource(event, persister, keyToLoad, options);
  22. }
protected Object doLoad(
		final LoadEvent event, 
		final EntityPersister persister,
		final EntityKey keyToLoad, 
		final LoadEventListener.LoadType options) 
	throws HibernateException {
		
		//1
		Object entity = loadFromSessionCache(event, keyToLoad, options);
		if ( entity == REMOVED_ENTITY_MARKER ) {
			return null;
		}
		if ( entity != null ) {
			return entity;
		}

		//2
		entity = loadFromSecondLevelCache(event, persister, options);
		if ( entity != null ) {
			return entity;
		}

		//3
		return loadFromDatasource(event, persister, keyToLoad, options);
	}

去掉其中的注释和logging部分,我们终于可以看到一个比较明显的语义了!!:::

1、loadFromSessionCache————从session缓存中查找, 如果找不到,则2

2、loadFromSecondLevelCache————从二级缓存中查找(如果支持),如果找不到,则3(即从sessionFactory查询)

3、loadFromDatasource————仍然查找不到,则从数据库中查询——这回是真正的查询

loadFromSessionCache 关键:

Object old = session.getEntityUsingInterceptor( keyToLoad );

loadFromSecondLevelCache关键:

Object ce = persister.getCache().get( ck, source.getTimestamp() );

loadFromDatasource关键:

Object entity = persister.load(
event.getEntityId(),
event.getInstanceToLoad(),
event.getLockMode(),
source
);

© 著作权归作者所有

共有 人打赏支持
村长大神
粉丝 162
博文 876
码字总数 904789
作品 0
杭州
程序员
Hibernate查询缓存全面分析

这里介绍Hibernate查询缓存对Iterator不起作用,只对List起作用。下面我们这种介绍把二级缓存 和 Hibernate查询缓存 结合使用。 AD: 本文向大家介绍Hibernate查询缓存,可能好多人还不了解H...

dong.li
2012/04/24
0
0
hibernate三级缓存概括

Hibernate三级缓存 一级缓存(session级别缓存): 1、session关闭后,缓存就没了;hibernate发出一条sql查询,若缓存中没有,则从数据库中查,查到之后放入缓存 2、以id为key值缓存对象 二级...

挨踢精英
2015/10/12
1K
0
hibernate笔记--缓存机制之 二级缓存(sessionFactory)和查询缓存

二级缓存(sessionFactory):   Hibernate的二级缓存由SessionFactory对象管理,是应用级别的缓存。它可以缓存整个应用的持久化对象,所以又称为“SessionFactory缓存”.   hibernate二级缓...

冬至饮雪
2016/03/09
0
0
java基础之框架hibernate

1.什么是缓存?  缓存是介于物理数据源与应用程序之间,是对数据库中的数据复制一份临时放在内存中的容器,其作用是为了减少应用程序对物理数据源访问的次数,从而提高了应用程序的运行性能...

sliver_xu
2016/08/10
7
0
在 Hibernate 中直接操作 JDBC 接口

简介: Hibernate 在处理多表关联及分组排序等复杂数据库查询操作时,其固有的 O-R 映射机制会产生大量冗余 SQL 操作,系统性能比传统的 JDBC 低很多。本文分析了 Hibernate 产生此类问题的原...

红薯
2010/04/16
639
2

没有更多内容

加载失败,请刷新页面

加载更多

下一页

RabbitMQ在CentOS环境安装

1.废话不多说准备一台虚拟机,系统为centos,我这里使用的系统版本如下图所示:

凌晨一点
45分钟前
0
0
线程池相关

在java.util.concurrent包下,提供了一系列与线程池相关的类。 使用线程池的好处 降低资源消耗。通过重复利用已创建的线程降低线程创建和销毁造成的消耗; 提高响应速度。当任务到达时,任务...

edwardGe
47分钟前
0
0
学习大数据这三个关键技术是一定要掌握!

大数据时代全面来临,大数据、人工智能等技术引领科技创新潮流,获得国家政策大力支持,前景广阔。学习大数据技术的人自然是络绎不绝, 学习大数据虽然是一个趋势,但也要注意大数据培训课程...

董黎明
今天
0
0
jetbrains 上传代码到github

设置中找github 获取token 验证是否成功 测试git 生成key,一路回车即可 ssh-keygen -t rsa -C “youremail@example.com” 打开pub复制key,需要再次输入一次密码 验证是否成功,输入yes即可...

阿豪boy
今天
0
0
分布式服务框架(拾遗)

前言 现在的大部分工程都已经是基于分布式架构来处理。所以这里对分布式框架做一个简单的总结 常用的RPC框架 RPC框架原理 RPC(Remote Procedure Call,远程过程调用)一般用来实现部署在不同...

kukudeku
今天
3
0

没有更多内容

加载失败,请刷新页面

加载更多

下一页

返回顶部
顶部