上一篇谈了我对缓存的概念,框架上的理解和看法,这篇承接上篇讲讲我自己的缓存模块设计实践。

基本的缓存模块设计

最基础的缓存模块一定有一个统一的CacheHelper,如下:

    public interface ICacheHelper
{
T Get<T>(string key); void Set<T>(string key, T value); void Remove(string key);
}

然后业务层是这样调用的

        public User Get(int id)
{
if (id <= )
throw new ArgumentNullException("id"); var key = string.Format(USER_CACHE_KEY, id);
var user = _cacheHelper.Get<User>(key);
if (user != null)
return user; return _repository.Get(id);
}

上面的代码没什么错误,但是实际运用的时候就产生疑问了,因为我一直强调缓存要保存"热数据",那样"热数据"一定会有过期的时候,我们不可能另外写一个去Set。所以干脆就结合到一起写是比较合适的。

public User GetV2(int id)
{
if (id <= )
throw new ArgumentNullException("id"); var key = string.Format(USER_CACHE_KEY, id);
var user = _cacheHelper.Get<User>(key);
if (user != null)
return user;

user = _repository.Get(id);
if (user != null)
_cacheHelper.Set(key, user);

return user;
}

上面的代码其实只是加了一个Set而已,就这样的设计的话,每次一个Get需要的重复代码实在是太多了,那么是不是应该更精简?这时候吃点C#语法糖就很有必要了,语法糖偶尔吃点增进效率,何乐而不为?

public User GetV3(int id)
{
if (id <= )
throw new ArgumentNullException("id"); var key = string.Format(USER_CACHE_KEY, id);
return _cacheHelperV2.Get<User>(key, () => _repository.Get(id));
} //ICache Get<T>实现
public T Get<T>(string key, Func<T> fetch = null)
{
T result = default(T);
var obj = Cache.Get(key);
if (obj is T)
{
result = (T)obj;
} if(result == null)
{
result = fetch(); if (result != null)
Set(key, result);
} return result;
}

这里我直接把Set方法都包装进了ICache.Get<T>,附带上Fetch Func。这样就把公共的操作抽象到了一起,简化了Cache的调用,完美的符合了我的想法。

缓存模块设计进阶

上一节里的ICache V3几乎已经最精简了,但是其实参考了ServiceStack.Redis之后,我发现了更加的抽象方式。很明显上一节的所有代码里,都是手动管理Key的,对于通常的对象Cache,这个Key还需要手动吗?来上最后一份改进。

public T Get<T>(object id, Func<T> fetch = null)
{
var type = typeof(T);
var key = string.Format("urn:{1}:{2}", type.Name, id.ToString());//这里是关键,直接用TypeName来充当Key return Get(key, fetch);
} public T Get<T>(string key, Func<T> fetch = null)
{
T result = default(T); var obj = Cache.Get(key);
if (obj is T)
{
result = (T)obj;
} if (result == null)
{
result = fetch(); if (result != null)
Set(key, result);
} return result;
}

Get方法完全自动化管理了Key,然后调用的方式再次被精简。

public User GetV4(int id)
{
if (id <= )
throw new ArgumentNullException("id"); return _cacheHelperV3.Get<User>(id, () => _repository.Get(id));
}

很明显还少了最重要的Set啊,Set的时候这个Key获取就要费一点事情了,最需要 解决的是如何获取这个主键id的值。

public class User
{
[PrimaryKey] //这个Attribute是最重要的东西
public int UserId { get; set;} public string UserName { get; set; } public string Cellphone { get; set; }
}
public void Set<T>(T obj)
{
//此处应该被缓存以提高反射的效率
var type = typeof(T);
var primaryKey = type.GetProperties()
.FirstOrDefault(t => t.GetCustomAttributes(false)
.Any(c => c is PrimaryKeyAttribute));//这里通过取PrimaryKeyAttribute来获取ID的value
var keyValue = primaryKey.GetValue(obj, null);
var key = string.Format("urn:{0}:{1}", type.Name, keyValue); var dt = DateTime.UtcNow.AddDays();//假设默认缓存1天
var offset = new DateTimeOffset(dt);
Cache.Set(key, obj, offset);
}

到这里,我想到的最终版本的ICache就完成了。这里还需要说明的是其实PrimaryKey可以更加灵活多变。很多时候一个Object的PrimaryKey是很复杂的,这时候设计Cache实体的时候可以变通下:

public class UserCacheEntity
{
[PrimaryKey]
public int ID
{
get
{
return string.Format("{0}:{1}", UserId, UserName);
}
} public int UserId { get; set; } public string UserName { get; set; } public string Cellphone { get; set; }
}

上面的方式几乎可以自动管理常见的数据Cache了,唯一麻烦的是 需要自定义一个CacheObject,这样就带来了实体转换的麻烦,这时候就要看怎么取舍了。

再次说明下我想要的ICache设计:

1. 永远只Cache热数据,这意味着每个Key都要有过期时间

2. ICache自动管理Get/Set,最好能自动管理Key。

3. ICache精简同时又不失灵活。

详细的代码Demo可以参考:Git

更灵活的实现

我在写这篇总结之前,也一直在思考Cache应该放到什么层,普通三层的时候放哪里?DDD那样分层的时候又放哪里。Google了下,看到了一些参考。

http://stackoverflow.com/questions/15340173/in-which-layer-implement-the-cache

我觉得这里比较符合我的想法,Cache应该是全局任意的,当然实现起来当然是interface+IOC,这样引用起来更加的独立一些。

另外还有Cache更加高级的使用,AOP结合ICache V4这样的设计,岂不是更好?这里我还没有去实现AOP的Attribute,这又是一个大话题的,下次再来实现吧。

本文比较粗陋,欢迎大家拍砖,期待共同进步。

.NET 缓存模块设计的更多相关文章

  1. IOS编程 图片缓存模块设计

    手机客户端为什么会留存下来?而不是被一味的Wap替代掉?因为手机客户端有Wap无可替代的优势,就是自身较强的计算能力. 手机中不可避免的一环:图片缓存,在软件的整个运行过程中显得尤为重要. 先简单说一 ...

  2. 解析大型.NET ERP系统 权限模块设计与实现

    权限模块是ERP系统的核心模块之一,完善的权限控制机制给系统增色不少.总结我接触过的权限模块,以享读者. 1 权限的简明定义 ERP权限管理用一句简单的话来说就是:谁 能否 做 那些 事. 文句 含义 ...

  3. 【Java EE 学习 78 上】【数据采集系统第十天】【Service使用Spring缓存模块】

    一.需求分析 调查问卷中或许每一个单击动作都会引发大量的数据库访问,特别是在参与调查的过程中,只是单击“上一页”或者“下一页”的按钮就会引发大量的查询,必须对这种问题进行优化才行.使用缓存策略进行查询 ...

  4. 利用Java的读写锁实现缓存的设计

    Java中的读写锁: 多个读锁不互斥, 读锁与写锁互斥, 写锁与写锁互斥, 这是由JVM自行控制的,我们只要上好相应的锁即可. 缓存的设计: package com.cn.gbx; import ja ...

  5. ylbtech-Model-Account(通用账户模块设计)

    ylbtech-DatabaseDesgin:ylbtech-Model-Account(通用账户模块设计) ylbtech-Model-Account(通用账户模块设计) 1.A,数据库关系图(Da ...

  6. atitit。浏览器缓存机制 and 微信浏览器防止缓存的设计 attilax 总结

    atitit.浏览器缓存机制 and 微信浏览器防止缓存的设计 attilax 总结 1. 缓存的一些机制 1 1.1. http 304 1 1.2. 浏览器刷新的处理机制 1 1.3. Expir ...

  7. ABP模块设计

    ABP模块设计 返回ABP系列 ABP是“ASP.NET Boilerplate Project (ASP.NET样板项目)”的简称. ASP.NET Boilerplate是一个用最佳实践和流行技术 ...

  8. asp.net通用查询模块设计

    asp.net通用查询模块设计 前言 自从上次狂喷了devexpress for asp.net面向互联网的app的各种不合理,好像骂的dev无处容身了,不过说实话,dev在做互联网的app时,生成的 ...

  9. thinkphp 5.0 模块设计

    模块设计 5.0版本对模块的功能做了灵活设计,默认采用多模块的架构,并且支持单一模块设计,所有模块的命名空间均以app作为根命名空间(可配置更改). 目录结构 标准的应用和模块目录结构如下: ├─ap ...

随机推荐

  1. HotSpot JVM常用参数设置

    转自:https://www.zybuluo.com/jewes/note/57352 选项的分类 Hotspot JVM提供以下三大类选项: 1. 标准选项:这类选项的功能是很稳定的,在后续版本中也 ...

  2. ADB

    adb logcat 命令详解 log过滤 http://blog.csdn.net/liao277218962/article/details/50129009 如何使用 adb logcat 查看 ...

  3. 【先定一个小目标】Windows下Redis的安装使用

    Redis是一个key-value存储系统.和Memcached类似,它支持存储的value类型相对更多,包括string(字符串).list(链表).set(集合).zset(sorted set ...

  4. 分享公司Basecode的用法

    主题 公司在basecode的用法上是比较有新意的,所以准备记录分享下公司的用法. 说明 basecode公司的一个主要用途就是用于一些基础的代码表,参数表的前台操作.这些表有很多,用spring d ...

  5. CSS常用技术总结!~~

    //放大屏幕,背景图不变 background: url(x.png) no-repeat 0 0; background-image: -webkit-image-set(url(logo_db.p ...

  6. 关于tkCommand的各种事件的解释

    superclass for callback/observer methods vtkCommand is an implementation of the observer/command des ...

  7. mysql基于“报错”的注入

    报错是如何转为xss的? mysql语句在页面报错,泄露信息 ===================================================================== ...

  8. MAC下Homebrew的安装

    1.Homebrew是啥东东? Homebrew的官方网站http://brew.sh/index.html上有这么一句“Homebrew installs the stuffyouneed that ...

  9. 常用shell 命令整理 一 进程 cpu

    1.查看内存从大到小排列 ps -e -o "%C : %p : %z : %a"|sort -k5 -nr 分析: -e 显示进程 -o 按用户自定义格式显示 %C cpu %p ...

  10. Sublime Text 3 Plugin Better!

    Package Control Cmake ConvertUTF Markdown preview MarkdownEditing Marking Changed Rows