hibernate二级缓存(二)二级缓存实现原理简单剖析

hibernate二级缓存(二)二级缓存实现原理简单剖析hibernate二级缓存(一)二级缓存的实现原理在前面我们将过hibernate二级缓存类似于一个插件,将缓存的具体实现分离,缓存的具体实现是通过hibernate.cache.region.factory_class参数配置指定。1.hibernate二级缓存结构hibernate二级缓存涉及到如下几个重要的接口:RegionFactoryDomainDataRegionEnt…

大家好,又见面了,我是你们的朋友全栈君。

hibernate二级缓存(二)二级缓存实现原理简单剖析

在前面我们将过hibernate二级缓存类似于一个插件,将缓存的具体实现分离,缓存的具体实现是通过hibernate.cache.region.factory_class参数配置指定。本文只是对hibernate二级缓存的部分接口进行简单的解析,大致了解二级缓存的整体结构,二级缓存的内部实现很复杂,如要深究请阅读hibernate源码。

1. hibernate二级缓存结构

hibernate二级缓存涉及到如下几个重要的接口:

  • RegionFactory
  • DomainDataRegion
  • EntityDataAccess
  • StorageAccess

1.1 RegionFactory获取缓存的工厂,RegionFactory有如下几个重要的方法:

public interface RegionFactory extends Service, Stoppable {
	//初始化方法
	void start(SessionFactoryOptions settings, Map configValues) throws CacheException;
	
	boolean isMinimalPutsEnabledByDefault();

	//缓存策略
	AccessType getDefaultAccessType();

	String qualify(String regionName);

	default CacheTransactionSynchronization createTransactionContext(SharedSessionContractImplementor session) {
		return new StandardCacheTransactionSynchronization( this );
	}
	
	long nextTimestamp();

	default long getTimeout() {
		// most existing providers defined this as 60 seconds.
		return 60000;
	}

	DomainDataRegion buildDomainDataRegion(
			DomainDataRegionConfig regionConfig,
			DomainDataRegionBuildingContext buildingContext);

	QueryResultsRegion buildQueryResultsRegion(String regionName, SessionFactoryImplementor sessionFactory);

	TimestampsRegion buildTimestampsRegion(String regionName, SessionFactoryImplementor sessionFactory);
}
  • start 初始化RegionFactory
  • getDefaultAccessType 获得缓存策略,前面说过的CacheConcurrencyStrategy内部是基于AccessType
  • nextTimestamp 生成时间戳,用于时间戳缓存
  • buildDomainDataRegion 创建一个实体领域模型的Region,使用该对象来缓存实体,可以理解为实体缓存的holder
  • buildQueryResultsRegion 创建查询缓存
  • buildTimestampsRegion 创建时间戳缓存。时间戳缓存Region存放了对于查询结果相关的表进行插入, 更新或删除操作的时间戳。Hibernate 通过时间戳缓存Region来判断被缓存的查询结果是否过期

RegionFactory 是创建缓存的工厂,所有的缓存都是通过RegionFactory 来获取的,而RegionFactory 是在EnabledCaching构造方法中初始化的。RegionFactory 的初始化过程如下图所示:
在这里插入图片描述
sessionFactory = new MetadataSources(registry).buildMetadata().buildSessionFactory();可以看到是在初始化sessionFactory的时候来初始化RegionFactory 。

1.2 DomainDataRegion缓存区域

DomainDataRegion可以理解为缓存的wraper或者holder。接口如下:

public interface DomainDataRegion extends Region {

    EntityDataAccess getEntityDataAccess(NavigableRole var1);

    NaturalIdDataAccess getNaturalIdDataAccess(NavigableRole var1);

    CollectionDataAccess getCollectionDataAccess(NavigableRole var1);
}

getEntityDataAccess 获得一个EntityDataAccess,对缓存的操作实际上是代理给EntityDataAccess,由EntityDataAccess来真正的管理缓存。

1.3 EntityDataAccess 缓存实际的访问者,用于管理对缓存实体数据的事务性和并发访问的协定

public interface EntityDataAccess extends CachedDomainDataAccess {
	//生成缓存key
	Object generateCacheKey(
			Object id,
			EntityPersister rootEntityDescriptor,
			SessionFactoryImplementor factory,
			String tenantIdentifier);
	//获取缓存key
	Object getCacheKeyId(Object cacheKey);
	//在查询后,是否将查询结果插入缓存
	boolean insert(SharedSessionContractImplementor session, Object key, Object value, Object version);
	//插入后是否更新缓存
	boolean afterInsert(SharedSessionContractImplementor session, Object key, Object value, Object version);
	
	boolean update(
			SharedSessionContractImplementor session,
			Object key,
			Object value,
			Object currentVersion,
			Object previousVersion);

	boolean afterUpdate(
			SharedSessionContractImplementor session,
			Object key,
			Object value,
			Object currentVersion,
			Object previousVersion,
			SoftLock lock);
}

在上面的接口中我们并没有看到实际操作缓存的接口,那么EntityDataAccess 又是怎么访问和管理缓存的呢,下面来看一下EntityDataAccess 的接口继承和实现关系:
在这里插入图片描述
从上面的图我们可以看到EntityDataAccess 有一个抽象类,4个实现类。4个实现类分别对应了4中缓存访问类型,READ_ONLY,TRANSACTIONAL,READ_WRITE,NONSTRICT_READ_WRITE。抽象类AbstractEntityDataAccess实现了
EntityDataAccess 继承了AbstractCachedDomainDataAccess。AbstractCachedDomainDataAccess的部分代码如下:

在这里插入图片描述
AbstractCachedDomainDataAccess里面包含了一个DomainDataStorageAccess,DomainDataStorageAccess继承自StorageAccess,StorageAccess的接口如下:

public interface StorageAccess {
	/**
	 * Get an item from the cache.
	 */
	Object getFromCache(Object key, SharedSessionContractImplementor session);

	/**
	 * Put an item into the cache
	 */
	void putIntoCache(Object key, Object value, SharedSessionContractImplementor session);

	/**
	 * Remove an item from the cache by key
	 */
	default void removeFromCache(Object key, SharedSessionContractImplementor session) {
		evictData( key );
	}

	/**
	 * Clear data from the cache
	 */
	default void clearCache(SharedSessionContractImplementor session) {
		evictData();
	}

	/**
	 * Does the cache contain this key?
	 */
	boolean contains(Object key);

	/**
	 * Clear all data regardless of transaction/locking
	 */
	void evictData();

	/**
	 * Remove the entry regardless of transaction/locking
	 */
	void evictData(Object key);

	/**
	 * Release any resources.  Called during cache shutdown
	 */
	void release();
}

到这里我们终于发现了操作缓存的实际接口。而DomainDataStorageAccess接口是缓存操作必须实现的接口,ehchche的实现类是StorageAccessImpl。当然通过扩展该接口我们还可以将缓存放置到redis,memcache。

2. 缓存的初始化和调用

2.1 从上面的接口关系我们大致可以得到如下的一个缓存初始化关系链:

sessionFactory —–>EnabledCaching —–>RegionFactory —–>DomainDataRegion —–>EntityDataAccess —–>AbstractEntityDataAccess —–>AbstractCachedDomainDataAccess —–>StorageAccess

2.2 获取cache的调用栈

在这里插入图片描述
SessionImpl获取实体对象,然后通过一系列的调用,最终会落到AbstractCachedDomainDataAccess.get(SharedSessionContractImplementor session, Object key),前面已经说过该方法实际上是调用的DomainDataStorageAccess.getFromCache( key, session )。

3. 自定义hibernate缓存

通过前面的一系列分析,我们大致了解了hibernate缓存的一些重要的接口。如果要自定义hibernate缓存那么我们必须实现上面的这些接口。好在hibernate内部为实现了大多数的扩展,我们只需要扩展RegionFactory和DomainDataStorageAccess接口既可以自定义hibernate的二级缓存。
hibernate为实现RegionFactory提供了一个模版类RegionFactoryTemplate,我们直接通过实现该类和DomainDataStorageAccess,即可自定义hibernate二级缓存。

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/142272.html原文链接:https://javaforall.net

(0)
全栈程序员-站长的头像全栈程序员-站长


相关推荐

  • 分享一个好用的Python在线编辑器

    分享一个好用的Python在线编辑器原文:https://lwebapp.com/zh/python-online需求有小伙伴可能听说过PyScript,知道了Python可以通过打包成wasm运行在浏览器端了,这样做一些需要Python来做的功能,可以直接在浏览器完成,无需和服务器交互,打开了开发者的想象力。这里我们要推荐的是一个在线工具,也是支持Python代码的执行,一个在线的Python代码编辑器Python在线编辑器地址:https://lwebapp.com/zh/python-playground如何使用Py

    2022年8月14日
    3
  • java 开发中 dom4j的简单用法「建议收藏」

    java 开发中 dom4j的简单用法「建议收藏」Java中处理XML的方式有很多种,个人任务dom4j还是比较好用的。下面介绍以下简单的使用方法先把import补充上1.首先我们一般会先根据各种方式得到一个xml格式的字符串或者文件,例如:

    2022年7月4日
    36
  • Python win32api_python api文档

    Python win32api_python api文档本文整理汇总了Python中win32api.SetCursorPos方法的典型用法代码示例。如果您正苦于以下问题:Pythonwin32api.SetCursorPos方法的具体用法?Pythonwin32api.SetCursorPos怎么用?Pythonwin32api.SetCursorPos使用的例子?那么恭喜您,这里精选的方法代码示例或许可以为您提供帮助。您也可以进一步了解该方…

    2022年10月11日
    1
  • 2020年3月25日阿里笔试题

    2020年3月25日阿里笔试题2020年3月25日阿里笔试题题目描述一python代码题目描述二求公差的python代码  仿佛人生总有一种魔咒,自己做的这场笔试题永远是最难的。不过今天的笔试题,真的难。来看题目。题目描述一给定一个数组n,然后给三个长度为n的数组,可以从这三个数组中选出一个长度为n的数组,第i个位置需要是从给出的三个数组第i个位置选择的,然后要求使这个数组后一项减前一项的绝对值之和最小。输入示例::…

    2022年5月24日
    35
  • 网络层传输层(计算机网络传输层大题)

    概述传输层的意义网络层可以把数据从一个主机传送到另一个主机,但是没有和进程建立联系。传输层就是讲进程和收到的数据联系到一起,使数据能够为应用服务所以说传输层是主机才有的层次传输层的两个协议传输层的寻址和端口端口号只用于计算机分辨本地进程,总共有2^16=65536种端口号,端口号有很多种,不能随便使用常见的应用程序端口号UDP协议UDP概述注释:因为UDP一次发送一个完整报文不会分片,所以需要应用层传输过来的数据不要太大,否则网络层分片任务就很重,但是也不能太小,不然效率较低

    2022年4月11日
    65
  • GROUP BY与COUNT用法详解

    GROUP BY与COUNT用法详解聚合函数在介绍GROUPBY和HAVING子句前,我们必需先讲讲sql语言中一种特殊的函数:聚合函数,例如SUM,COUNT,MAX,AVG等。这些函数和其它函数的根本区别就是它们一般作用在多条记录上。SELECTSUM(population)FROMbbc这里的SUM作用在所有返回记录的population字段上,结果就是该查询只返回一个结果,即国家的总人口数。

    2022年5月9日
    40

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号