采用EntLib5.0(Unity+Interception+Caching)实现项目中可用的Caching机制
看了园子里很多介绍Caching的文章,多数都只介绍基本机制,对于Cache更新和依赖部分,更是只简单的实现ICacheItemRefreshAction接口,这在实际项目中是远远不够的。实际项目中,至少应该考虑以下3点:
- 外部数据:通过外部服务,从其他系统取来的数据。我们无法控制,也不知道啥时候会被更新。对于这部分数据,我们采用定时更新的策略,默认1小时更新1次,可配置。
- 内部数据:系统自己产生的数据,可以完全掌控。对这部分数据,我们实现ICacheItemRefreshAction接口,一旦失效,立即重取。
- 内部数据如何触发更新:具体哪些操作会触发哪个cache失效,例如Add/Delete/Update[User][s]->Get[User][s]失效,这种策略应该可以配置。
接下去分别展开。
0、前言
下面的讨论分成2大块,一是环境的搭建,包括用Unity提供服务实例+用Unity提供Interception。经过这一块,就可以AOP的方式实现Caching了,这部分园子里的文章很多,我简略的过一下。二是Caching的实现,分别从外部数据缓存+内部缓存+内部触发更新3部分来讨论。
1、环境的搭建
1.1、用Unity提供服务实例
这部分其实就是用Unity实现IInstanceProvider,再配合IEndpointBehavior之类的,来实现WCF与Unity的集成。核心代码如下:
public class UnityInstanceProvider : IInstanceProvider{
public object GetInstance(InstanceContext context, Message msg){
return UnityWrapper.Instance.Resolve(serviceType);
}
}
具体可参见Artech的这篇《WCF后续之旅(7):通过WCF Extension实现和Enterprise Library Unity Container的集成》。
1.2、用Unity提供Interception(即PIAB,PolicyInjection)
这里要先吐槽一下EntLib,向后的兼容性做的太差了,完全不符合微软一贯的风格。每出一个大版本就一堆Breaking Changes和Deprecated,真心有点郁闷。当然估计他们也有苦衷,比如EntLib6里的Caching,被包含到Net40里了,确实也只能放弃。好吧,回归正题。好在EntLib自带的帮助文档写的非常详尽,有问题应该优先去Manual里查看相关内容。
EntLib4.1的PIAB模块,到EntLib5里就完全基于Unity的Interception来实现了,到EntLib6里就干脆取消了。这里我们采用EntLib5里的配置来实现策略注入,核心配置如下:
<unity>
<sectionExtension type="Microsoft.Practices.Unity.InterceptionExtension.Configuration.InterceptionConfigurationExtension, Microsoft.Practices.Unity.Interception.Configuration"/>
<assembly name="Service"/>
<namespace name="Service"/>
<alias alias="CachingCallHandler" type="Service.Unity.CachingCallHandler, Service"/>
<container name="UnityContainer">
<extension type="Interception" />
<interception>
<policy name="caching">
<matchingRule name="allServiceMatch" type="NamespaceMatchingRule">
<constructor>
<param name="namespaceName" value="Service"/>
</constructor>
</matchingRule>
<callHandler name="cachingService" type="CachingCallHandler"/>
</policy>
</interception>
<register type="IService1">
<interceptor type="TransparentProxyInterceptor" />
<policyInjection />
</register>
</container>
</unity>
这里有个细节,为了方便配置,1是可以用EntLib里自带的EntLibConfig.exe,2是在vs菜单->XML->架构里,添加UnityConfiguration20.xsd或者EnterpriseLibrary.Configuration.xsd,这样就有智能提示了。如果有EntLib的安装目录里找不到.xsd文件,可以用Everything搜一下。
2、Caching的实现
2.1、外部数据缓存
外部数据是我们不可控的数据,什么时候更新不确定。我们采取定时更新的策略,更新的频率主要取决于外部数据的变化频率,比如我们项目里依赖是外部的基础数据,变化较少,所以只需1小时甚至半天更新一次。
原本我以为AbsoluteTime是我想要的(注:SlidingTime是用户第一次命中Cache后,再滑动一段时间才过期),但通过ILSpy查看内部代码才知道,EntLib不是在一到期就自动触发ICacheItemRefreshAction接口的,而是在用户访问Cache的时候,再根据时间判断是否过期,如果过期再触发RefreshAction。代码如下:
namespace Microsoft.Practices.EnterpriseLibrary.Caching{
public class Cache{
public object GetData(string key){
if(cacheItem.HasExpired()){
this.inMemoryCache.Remove(key);
RefreshActionInvoker.InvokeRefreshAction(cacheItem, CacheItemRemovedReason.Expired, this.instrumentationProvider);
this.instrumentationProvider.FireCacheExpired(1L);
}}}}
这样的用户体验当然差了些,不能让用户享受到Caching的全部好处,我们改用Timer自动更新Cache,注意请使用System.Threading.Timer,而不是System.Timers.Timer。代码如下://我们用自定义的TimerCacheAttribute来标识这类方法
var atts = input.MethodBase.GetCustomAttributes(typeof(TimerCacheAttribute), false);
if(atts.Length > ){
RefreshTimerUtil.Instance[key] = new Timer(e => {
object[] args = e as object[];
if(args == null || args.Length != ) return; var tmpInput = args[] as IMethodInvocation;
var tmpNext = args[] as GetNextHandlerDelegate;
var tmpReturn = tmpNext()(tmpInput, tmpNext);
if(tmpReturn.Exception == null) {
CacheWrapper.Instance.Add(key, tmpReturn.ReturnValue);
}
},
new object[] { input, getNext },
TimeSpan.Zero,
new TimeSpan(, , ));
}
2.2、内部数据缓存
内部数据可以直截了当的插入Cache,然后就静静的等待增删改操作来把它置为过期。
CacheWrapper.Instance.Add(key, methodReturn.ReturnValue,
CacheItemPriority.Normal,
new CacheRefreshAction { Input = input, GetNext = getNext });
当然,要实现ICacheItemRefreshAction接口,代码如下:
[Serializable]
public class CacheRefreshAction : ICacheItemRefreshAction {
public IMethodInvocation Input {get; set;}
public GetNextHandlerDelegate GetNext {get; set;}
public void Refresh(string removedKey, object expiredValue, CacheItemRemovedReason removalReason) {
if( Input == null || GetNext == null) return;
var methodReturn = GetNext()(Input, GetNext);
if(methodReturn.Exception == null) {
CacheWrapper.Instance.Add(removedKey, methodReturn.ReturnValue,
CacheItemPriority.Normal,
new CacheRefreshAction { Input = Input, GetNext = GetNext });
}
}
}
据说MemoryCache是不需要实现ISerializable的,而其他Caching要么实现ISerializable、要么打[Serializable]标签。不过这种实现方式,IMethodInvocation和GetNextHandlerDelegate目测序列化都略麻烦,好在目前我们只用到MemoryCache,可以先不管,以后再突破。
2.3、内部数据触发更新
最后就剩触发更新了,如何实现类似Add/Delete/Update[User][s]->Get[User][s]的效果呢?大致思路是:1)判断方法名里是否StartWith关键字(Add/Delete/Update);2)截取出目标关键字User、并生成对应的复数形式Users;3)与Get等动作拼接成CacheKey,把所有命中的CacheKey置为过期(CacheManager.Remove),即可触发对应的RefreshAction。代码如下:
var entity = string.Empty;
foreach (var act in ConfigUtil.Action){
if(methodName.StartsWith(act)) {
entity = methodName.Substring(act.Length);
break;
}} var entities = PluralUtil.Pluralize(entity);
if(entities == entity) entities = PluralUtil.Singularize(entity);
foreach (var prefix in ConfigUtil.Prefix) {
CacheWrapper.Instance.RemoveStartWith(prefix + entity);
CacheWrapper.Instance.RemoveStartWith(prefix + entities);
}
这里的产生单复数的代码是从EntityFramework里借来的,具体请查看EnglishPluralizationService。
如果有不正确的地方,欢迎批评指正!
采用EntLib5.0(Unity+Interception+Caching)实现项目中可用的Caching机制的更多相关文章
- .NET Core 3.0或3.1 类库项目中引用 Microsoft.AspNetCore.App
本文为原创文章.首发:http://www.zyiz.net/ 在 ASP.NET Core 3.0+ web 项目中已经不需要在 .csproj 中添加对 Microsoft.AspNetCore. ...
- 分享我们项目中基于EF事务机制的架构
写在前面: 1. 本文中单元测试用到的数据库,在执行测试之前,会被清空,即使用空数据库. 2. 本文中的单元测试都是正确通过的. 要理解EF的事务机制,首先要理解这2个类:TransactionSco ...
- 分享我们项目中基于EF事务机制的架构 【转载】
http://www.cnblogs.com/leotsai/p/how-to-use-entity-framework-transaction-scope.html 写在前面: 1. 本文中单元测试 ...
- Python:Django 项目中可用的各种装备和辅助
1 Redis 数据库 2 MySQL 数据库 3 前端服务器 live-server 4 定时任务 django-crontab扩展 5 Docker 容器 --用来运行 FastDFS 分布式文件 ...
- DotNet项目中的一些常用验证操作
在项目中需要对用户输入的信息,以及一些方法生成的结果进行验证,一般在项目中较多的采用js插件或js来进行有关信息的校验,但是从项目安全性的角度进行考虑,可对系统进行js注入. 如果在后台对用户输入的信 ...
- 在 ASP.NET Core 项目中使用 npm 管理你的前端组件包
一.前言 在项目的前端开发中,对于绝大多数的小伙伴来说,当然,也包括我,不可避免的需要在项目中使用到一些第三方的组件包.这时,团队中的小伙伴是选择直接去组件的官网上下载,还是图省事直接在网上搜索,然后 ...
- 如何在Crystal框架项目中内置启动MetaQ服务?
当Crystal框架项目中需要使用消息机制,而项目规模不大.性能要求不高时,可内置启动MetaQ服务器. 分步指南 项目引入crystal-extend-metaq模块,如下: <depende ...
- 1-跑Faster R-CNN项目中的一些问题
原理介绍: https://blog.csdn.net/quincuntial/article/details/79132243 我用的环境: Python 3.5.2 cpu版的ten ...
- java项目中常见的异常及处理
Java开发中常见异常及处理方法 1.JAVA异常 异常指不期而至的各种状况,如:文件找不到.网络连接失败.非法参数等.异常是一个事件,它发生在程序运行期间,干扰了正常的指令流程.Java通 过API ...
随机推荐
- Path,Files巩固,题目:从键盘接收两个文件夹路径,把其中一个文件夹中(包含内容)拷贝到另一个文件夹中
这个题目用传统的File,InputStream可以做,但是如果用Files,Path类做,虽然思路上会困难一些,但是代码简洁了很多,以下是代码: import java.io.IOException ...
- Android_用户界面概述和数据单位
一.UI界面概述 UI,对于一个应用而言用户界面是非常重要的一部分,是应用的脸,用户对应用第一个印象来自于界面,因此如果没有完美的用户界面,很难留住用户. 好的用户界面会极大提高用户的使用欲望并维护客 ...
- 通过NORFLASH中的uboot烧写uboot到nandFlash
在mini2440的教程中,在构建nandflash系统的时候是首先通过supervivi借助dnw烧写uboot.bin到nand flash 第零块, 由于我使用的是64位操作系统,usb驱动没安 ...
- 十问 Linux 虚拟内存管理 (glibc) (二)
版权声明:本文由陈福荣原创文章,转载请注明出处: 文章原文链接:https://www.qcloud.com/community/article/184 来源:腾云阁 https://www.qclo ...
- noip赛前小结4
真正的勇士,敢于面对惨淡的人生. 真正的OIer,敢于做ccop的题. 有种凄凉叫做这道数学题已经超出了我语文的理解范围. 有种愤怒叫做ccop类的信息题已经超出了我语文的理解范围和数学的理解范围. ...
- 配置electron
配置语句: git clone https://github.com/electron/electron-quick-start 文件夹名字 打开该文件(我用的webstorm)
- JAVA中线程同步的方法(7种)汇总
同步的方法: 一.同步方法 即有synchronized关键字修饰的方法. 由于java的每个对象都有一个内置锁,当用此关键字修饰方法时, 内置锁会保护整个方法.在调用该方法前,需要获得内置锁,否则就 ...
- Merkle Tree学习
/*最近在看Ethereum,其中一个重要的概念是Merkle Tree,以前从来没有听说过,所以查了些资料,学习了Merkle Tree的知识,因为接触时间不长,对Merkle Tree的理解也不是 ...
- hdu-----(4857)逃生(拓扑排序)
逃生 Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others)Total Submissi ...
- javaSE基础之记事本编程
首先安装好jdk和jre,之后进行如下操作: 1. 将代码记事本---->cmd--->javac 文件名.java ----->java 文件名 如图: 2. 关于记事本文件属性的 ...