前言

以前整理过缓存的东西在:

https://www.cnblogs.com/aoximin/p/12727659.html

只是粗略的例子,因为真的要去介绍缓存这个东西,要从内存开始,是一个有时间系列。

该文通过分析盛派源码,简单介绍如何实现一个简单的缓存机制。

正文

查看源码的出发点,我当时是这样想的,无论你采用哪种缓存,那么你都得暴露出一个object,让我可以进程增删改查吧。

在BaseContainer中,查看到:

/// <summary>
/// 获取符合当前缓存策略配置的缓存的操作对象实例
/// </summary>
protected static IBaseObjectCacheStrategy /*IBaseCacheStrategy<string,Dictionary<string, TBag>>*/ Cache
{
get
{
//使用工厂模式或者配置进行动态加载
//return CacheStrategyFactory.GetContainerCacheStrategyInstance(); //以下代码可以实现缓存“热切换”,损失的效率有限。如果需要追求极致效率,可以禁用type的判断
var containerCacheStrategy = ContainerCacheStrategyFactory.GetContainerCacheStrategyInstance()/*.ContainerCacheStrategy*/;
if (_containerCache == null || _containerCache.GetType() != containerCacheStrategy.GetType())
{
_containerCache = containerCacheStrategy;
} if (_baseCache == null)
{
_baseCache = _baseCache ?? containerCacheStrategy.BaseCacheStrategy();
}
return _baseCache;
}
}

像这种属性,不用看肯定是单例了。

ContainerCacheStrategyFactory 是一个生产者,那么看下它到底生成了啥吧。

GetContainerCacheStrategyInstance:

public static IContainerCacheStrategy GetContainerCacheStrategyInstance()
{
//从底层进行判断
var containerCacheStrategy = CacheStrategyFactory.GetExtensionCacheStrategyInstance(ContainerCacheStrategyDomain.Instance)
as IContainerCacheStrategy;
return containerCacheStrategy;
}

ContainerCacheStrategyFactory 相当于是一片生产园container,CacheStrategyFactory是园区里面得工厂,建筑学。

好的,那么就看一下CacheStrategyFactory这个工程到底生产了啥吧。

查看:ContainerCacheStrategyDomain.Instance

public class ContainerCacheStrategyDomain : ICacheStrategyDomain
{ #region 单例 /// <summary>
/// LocalCacheStrategy的构造函数
/// </summary>
ContainerCacheStrategyDomain() : base()
{
} //静态LocalCacheStrategy
public static ICacheStrategyDomain Instance
{
get
{
return Nested.instance;//返回Nested类中的静态成员instance
}
} class Nested
{
static Nested()
{
}
//将instance设为一个初始化的LocalCacheStrategy新实例
internal static readonly ContainerCacheStrategyDomain instance = new ContainerCacheStrategyDomain();
} #endregion private const string IDENTITY_NAME= "6526BBC0-718A-4F47-9675-D6DF6E1CE125";//固定值,请勿修改
private const string DOMAIN_NAME = "WeixinContainer";//固定值,请勿修改。同时会作为缓存键命名空间的子级名称 public string IdentityName { get { return IDENTITY_NAME; } } public string DomainName { get { return DOMAIN_NAME; } }
}

里面是一些固定值:

private const string IDENTITY_NAME= "6526BBC0-718A-4F47-9675-D6DF6E1CE125";//固定值,请勿修改
private const string DOMAIN_NAME = "WeixinContainer";//固定值,请勿修改。同时会作为缓存键命名空间的子级名称

看到这种情况,可以想象到基本是和注册有关(注册服务)。

然后进GetExtensionCacheStrategyInstance看下:

/// <summary>
/// 获取指定领域缓存的换存策略
/// </summary>
/// <param name="cacheStrategyDomain">领域缓存信息(需要为单例)CacheStrategyDomain</param>
/// <returns></returns>
public static IDomainExtensionCacheStrategy GetExtensionCacheStrategyInstance(ICacheStrategyDomain cacheStrategyDomain)
{
return CacheStrategyDomainWarehouse.GetDomainExtensionCacheStrategy(GetObjectCacheStrategyInstance(), cacheStrategyDomain);
}

cacheStrategyDomain 是我们传进来得是一个实体类,后续猜测是一个注册标识对象。

GetObjectCacheStrategyInstance 看下这个干啥了。

/// <summary>
/// 注册当前全局环境下的缓存策略,并立即启用。
/// </summary>
/// <param name="func">如果为 null,将使用默认的本地缓存策略(LocalObjectCacheStrategy.Instance)</param>
public static void RegisterObjectCacheStrategy(Func<IBaseObjectCacheStrategy> func)
{
ObjectCacheStrateFunc = func;
if (func != null)
{
ObjectCacheStrategy = func();
}
} /// <summary>
/// 获取全局缓存策略
/// </summary>
/// <returns></returns>
public static IBaseObjectCacheStrategy GetObjectCacheStrategyInstance()
{
if (ObjectCacheStrateFunc == null)
{
return LocalObjectCacheStrategy.Instance;
}
return ObjectCacheStrateFunc();
}

上面RegisterObjectCacheStrategy表示我们可以注册一种缓存策略,ObjectCacheStrateFunc 将会保存。

GetObjectCacheStrategyInstance 就是我们需要看的,如果我们没有注册得话,那么他就会用本地的缓存,LocalObjectCacheStrategy.Instance。

查看LocalObjectCacheStrategy.Instance是什么?

里面是一堆方法,设置key value 过期时间等。不过我找到了这个:

static LocalObjectCacheHelper()
{
LocalObjectCache = HttpRuntime.Cache;
}

默认使用的是:HttpRuntime.Cache;

好的,那么参数我们都知道了,那么看下return CacheStrategyDomainWarehouse.GetDomainExtensionCacheStrategy(GetObjectCacheStrategyInstance(), cacheStrategyDomain);中的GetDomainExtensionCacheStrategy干了啥吧。

/// <summary>
/// 获取领域缓存(指定特定 的IBaseObjectCacheStrategy 缓存策略对象)
/// </summary>
/// <param name="baseObjectCacheStrategy">IBaseObjectCacheStrategy 缓存策略对象</param>
/// <param name="cacheStrategyDomain">缓存领域</param>
/// <returns></returns>
public static IDomainExtensionCacheStrategy GetDomainExtensionCacheStrategy(IBaseObjectCacheStrategy baseObjectCacheStrategy, ICacheStrategyDomain cacheStrategyDomain)
{
CacheStrategyDomainMappingCollection mappingCollection = GetMappingCollection(cacheStrategyDomain.IdentityName);
if (mappingCollection.ContainsKey(baseObjectCacheStrategy))
{
return ((Dictionary<IBaseObjectCacheStrategy, CacheStrategyDomainMappingItem>)mappingCollection)[baseObjectCacheStrategy].DomainExtensionCacheStrategy;
}
UnregisteredDomainCacheStrategyException ex = new UnregisteredDomainCacheStrategyException(cacheStrategyDomain.GetType(), baseObjectCacheStrategy.GetType());
SenparcTrace.BaseExceptionLog(ex);
throw ex;
}

上面的从代码表面意思就是去一个映射集合中更具identityName查找,如果没有的话就抛出异常,如果有的话,就返回注册过的对象。

/// <summary>
/// 获取某个领域内的所有CacheStrategyDomainMappingCollection
/// </summary>
/// <param name="identityName"></param>
/// <returns></returns>
private static CacheStrategyDomainMappingCollection GetMappingCollection(string identityName)
{
if (!_extensionCacheStrategyInstance.ContainsKey(identityName))
{
_extensionCacheStrategyInstance[identityName] = new CacheStrategyDomainMappingCollection();
}
return _extensionCacheStrategyInstance[identityName];
}

好吧,既然是这样那么肯定有注册啊。

找到注册:

/// <summary>
/// 注册领域缓存
/// </summary>
/// <param name="domainCacheStrategy"></param>
public static void RegisterCacheStrategyDomain(IDomainExtensionCacheStrategy domainCacheStrategy)
{
string identityName = domainCacheStrategy.CacheStrategyDomain.IdentityName;
IBaseObjectCacheStrategy key = domainCacheStrategy.BaseCacheStrategy();
CacheStrategyDomainMappingCollection mappingCollection = GetMappingCollection(identityName);
CacheStrategyDomainMappingItem cacheStrategyDomainMappingItem2 = ((Dictionary<IBaseObjectCacheStrategy, CacheStrategyDomainMappingItem>)mappingCollection)[key] = new CacheStrategyDomainMappingItem(domainCacheStrategy);
}

这里IDomainExtensionCacheStrategy 里面其实就是IBaseObjectCacheStrategy baseObjectCacheStrategy, ICacheStrategyDomain cacheStrategyDomain。

public interface IDomainExtensionCacheStrategy
{
/// <summary>
/// 领域缓存定义
/// </summary>
ICacheStrategyDomain CacheStrategyDomain
{
get;
} /// <summary>
/// 使用的基础缓存策略
/// </summary>
Func<IBaseObjectCacheStrategy> BaseCacheStrategy
{
get;
} /// <summary>
/// 向底层缓存注册当前缓存策略
/// </summary>
/// <param name="extensionCacheStrategy">扩展缓存策略实例</param>
void RegisterCacheStrategyDomain(IDomainExtensionCacheStrategy extensionCacheStrategy);
}

其实后面就是一个ioc了。

因为很多人没有去了解这个框架,也不需要了解这个框架,所以就不会发布到首页中。

重学c#系列——缓存[盛派源码分析cache](九)的更多相关文章

  1. Java入门系列之集合LinkedList源码分析(九)

    前言 上一节我们手写实现了单链表和双链表,本节我们来看看源码是如何实现的并且对比手动实现有哪些可优化的地方. LinkedList源码分析 通过上一节我们对双链表原理的讲解,同时我们对照如下图也可知道 ...

  2. 一步步实现windows版ijkplayer系列文章之六——SDL2源码分析之OpenGL ES在windows上的渲染过程

    一步步实现windows版ijkplayer系列文章之一--Windows10平台编译ffmpeg 4.0.2,生成ffplay 一步步实现windows版ijkplayer系列文章之二--Ijkpl ...

  3. Volley源码解析(三) 有缓存机制的情况走缓存请求的源码分析

    Volley源码解析(三) 有缓存机制的情况走缓存请求的源码分析 Volley之所以高效好用,一个在于请求重试策略,一个就在于请求结果缓存. 通过上一篇文章http://www.cnblogs.com ...

  4. Redisson分布式锁学习总结:可重入锁 RedissonLock#lock 获取锁源码分析

    原文:Redisson分布式锁学习总结:可重入锁 RedissonLock#lock 获取锁源码分析 一.RedissonLock#lock 源码分析 1.根据锁key计算出 slot,一个slot对 ...

  5. Java入门系列之集合HashMap源码分析(十四)

    前言 我们知道在Java 8中对于HashMap引入了红黑树从而提高操作性能,由于在上一节我们已经通过图解方式分析了红黑树原理,所以在接下来我们将更多精力投入到解析原理而不是算法本身,HashMap在 ...

  6. Java入门系列之集合ArrayList源码分析(七)

    前言 上一节我们通过排队类实现了类似ArrayList基本功能,当然还有很多欠缺考虑,只是为了我们学习集合而准备来着,本节我们来看看ArrayList源码中对于常用操作方法是如何进行的,请往下看. A ...

  7. Flask系列10-- Flask请求上下文源码分析

    总览 一.基础准备. 1. local类 对于一个类,实例化得到它的对象后,如果开启多个线程对它的属性进行操作,会发现数据时不安全的 import time from threading import ...

  8. SSO单点登录系列1:cas客户端源码分析cas-client-java-2.1.1.jar

    落雨 cas 单点登录 希望能给以后来研究cas的兄弟留下一点思路,也算是研究了两天的成果,外国人的代码写的很晦涩,翻译下来也没有时间继续跟进,所以有错误的还请大家跟帖和我讨论,qq 39426378 ...

  9. Java入门系列之集合Hashtable源码分析(十一)

    前言 上一节我们实现了散列算法并对冲突解决我们使用了开放地址法和链地址法两种方式,本节我们来详细分析源码,看看源码中对于冲突是使用的哪一种方式以及对比我们所实现的,有哪些可以进行改造的地方. Hash ...

  10. 源码分析系列1:HashMap源码分析(基于JDK1.8)

    1.HashMap的底层实现图示 如上图所示: HashMap底层是由  数组+(链表)+(红黑树) 组成,每个存储在HashMap中的键值对都存放在一个Node节点之中,其中包含了Key-Value ...

随机推荐

  1. Java 类中属性的使用

    1 类中属性的使用: 2 * 属性(成员变量) 局部变量 3 * 1.相同点: 4 * 定义变量的个格式: 数据类型 变量名 = 变量值 5 * 先声明 后使用 6 * 变量都有其对应的作用域 7 * ...

  2. JAVA 多线程---面经

    线程与进程 提到进程那就要说程序,程序有指令和数据,程序从磁盘加载到内存,cpu获得指令进行执行,其中还会用到各种资源如网络资源,磁盘等.一个程序从磁盘进入内存,就是进程实例的创建. 一个程序可以有多 ...

  3. .bat 批处理 手册 教程

    有时候bat写个脚本 还是挺方便的,网上也没有不错的手册,有时间再整理看吧.网上找几个,先留存. 系统变量 %USERPROFILE% https://blog.csdn.net/ztx114/art ...

  4. Ambiq Micro-AMA3B2KK-KBR芯片可穿戴产品解决方案开发之基于Freertos系统的按键和LED控制

    一 前记 freetos在可穿戴设备上应用十分广泛,用来实现基本的按键和led控制,来熟悉它的基本架构.这是一个不错的尝试. 二 源码分析 1 系统初始化 这个启动task,相当于系统的入口函数,这里 ...

  5. linux中ping命令停不下来解决方案

    linux的 ping 命令和windows不一样.windows默认只发送四个包的. 你可以使用ping -c 4 [ip/域名]这种方式来实现你想要的. linux控制台程序一般强制终止都是Ctr ...

  6. 大年学习linux(第四节---文件权限)

    四.文件权限 文件类型 Linux文件类型和linux文件的文件名所代表的意义是两个不同的概念.我们通过一般应用程序而创建的比如 file.txt.file.tar.gz ,这些文件虽然要用不同的程序 ...

  7. 基于Rust的Tile-Based游戏开发杂记(01)导入

    什么是Tile-Based游戏? Tile-based游戏是一种使用tile(译为:瓦片,瓷砖)作为基本构建单位来设计游戏关卡.地图或其他视觉元素的游戏类型.在这样的游戏中,游戏世界的背景.地形.环境 ...

  8. 从null-ls归档再看nvim的代码格式化与lint方案

    由于null-lsp的归档和暂停更新,我们需要重新审视并思考还有哪些架构简单易于理解的插件配置方案.本文将介绍脱离null-ls插件体系下的代码格式化和lint的插件配置方案. 在之前的文章中< ...

  9. vue初学核心基础02

    8.v-bind补充 8.1v-bind绑定类名 v-bind指令给"任意标签"的"任意属性"绑定数据 对于大部分的属性而言我们只需要直接赋值即可, 例如:va ...

  10. 立创EDA的使用

    立创EDA的使用 1.实验原理 最近在使用立创EDA来做电路作业,这里记录一下立创EDA的基本操作,以后小型的电路设计可以在其主页完成.立创EDA是一个可以线上完成电路设计仿真以及布线的免费设计工具, ...