缓存键的设计

本贴最后更新于 1753 天前,其中的信息可能已经东海扬尘

很多过度设计(overengineering)借着柔性设计的名义而自认为是正当的。但是,过多的抽象层和间接设计常常成为项目的绊脚石。看一下真正为用户带来强大功能的软件设计,你会发现他们通常有一些非常简单的部分。简单并不容易做到。

                                                                                                                   ---来自 Eric Evans《领域驱动设计》    

 上面的引文当然和正文无关,对领域驱动也是了解甚少。偶然读到的,感觉挺有道理,就装 B 引用一下,下面开始正文。

如果在系统中使用过缓存,肯定会意识到有“缓存键”这么一个概念,不管是 memcached 还是 redis 都是以字符串作为缓存键的。我要说的这个缓存键设计是在我们的系统中以什么样的方式得到这个字符串。

可能有些人会说,直接以字符串作为缓存键不就可以了吗?直接用字符串肯定是可以的,但是维护性不太好,缓存键可能遍布整个系统,就算在一个地方维护所有的键,使用者也可以随意传参,比如:

    class StringCacheKeys
    { public static readonly string SystemName = "SystemName"; public static readonly string NewsDetails = "NewsDetails_{0}";
    } class AppStringCache
    { public static object GetValue(string key)
        { return null;
        } public static void Invoke()
        {
            GetValue(StringCacheKeys.SystemName);
            GetValue(string.Format(StringCacheKeys.NewsDetails, 23));
            GetValue("sbadsfsdf");
        }
    }

如上代码,GetValue 方法是使用缓存的方法,参数按我们假设用 string 类型,在 Invoke 方法里,可以传入任何字符串,虽然保证了灵活性,但失去了规范。

也许有人会用枚举来作为缓存键,单独使用枚举,肯定是很规范的,但是灵活性就不行了,很多时候缓存键都需要额外的具体参数填充才行,比如上面的 NewsDetails_{0},我们期望根据新闻编号来缓存新闻,所以使用枚举的话,必定要借助其他的手段才能实现灵活性,比如特性(Attribute):

 [AttributeUsage(AttributeTargets.Field)] class EnumCacheKeyDescriptorAttribute : Attribute
    { public string Key { get; set; } public EnumCacheKeyDescriptorAttribute(string key)
        {
            Key = key;
        }
    } enum EnumCacheKey
    {
        [EnumCacheKeyDescriptor("SystemName")]
        SystemName,

        [EnumCacheKeyDescriptor("NewsDetails_{0}")]
        NewsDetails,
    } class AppEnumCacheKey
    { public static object GetValue(EnumCacheKey key)
        { return null;
        } public static object GetValue(EnumCacheKey key, params object[] args)
        { var format = ""; //取出EnumCacheKeyDescriptor.Key;
            var realKey = string.Format(format, args); return null;
        }
    }

虽然可以解决问题,但是现在使用缓存的接口已经是两个了,一个没有附加参数,一个有附加参数,感觉还是不好。

所以还是求助于类,求助于面向对象:

    public class CacheKey
    {
        TimeSpan _expires; string _key; public CacheKey(string key, TimeSpan expires)
        {
            _key = key;
            _expires = expires;
        } public TimeSpan GetExpires()
        { return _expires;
        } public virtual string GetKey()
        { if (_key.IndexOf("{0}") >= 0)
            { throw new Exception(_key + "需要额外参数,请调用BuildWithParams设置");
            } return _key;
        } public CacheKey BuildWithParams(params object[] args)
        { if (args.Length == 0)
            { throw new Exception("如果没有参数,请不要调用BuildWithParams");
            } var m = new ParamsCacheKey(_key, _expires, args); return m;
        } class ParamsCacheKey : CacheKey
        { object[] _args; public ParamsCacheKey(string key, TimeSpan expires, object[] args) : base(key, expires)
            {
                _args = args;
            } public override string GetKey()
            { return string.Format(_key, _args);
            }
        }
    }

如此这般的设计一番,是否满足了我们需求呢?第一,使用缓存的接口统一为 CacheKey,第二,如果需要参数,在使用的时候需要调用一下 BuildWithParams 方法,该方法生产一个 CacheKey 的不公开子类 ParamsCacheKey 并返回,这个 ParamsCacheKey 负责参数的处理。代码中还有两处抛出异常的代码,异常应该就是在这种情况下使用的吧!我们订制了规则而调用者不按照规则使用,当然要回复以异常了。我们可以像上面一样定义一个 CacheKeys 来统一维护缓存键:

    public static class CacheKeys
    { public static CacheKey NameCacheKey = new CacheKey("Name", TimeSpan.FromHours(1)); public static CacheKey NewsCacheKey = new CacheKey("News_{0}", TimeSpan.FromHours(1));  }

CacheKey 到此结束!

那么有参数的缓存键和无参数的缓存键到底有什么区别呢?不知道大家在思考这个问题的时候能想到什么,我当时是用这个问题驱动我的思维的。之后还想到的两个相关的概念:

第一个是装饰器模式(允许向一个现有的对象添加新的功能,同时又不改变其结构)。我们用装饰器模式可以这样实现:

    class ThinkDecoration
    { abstract class CacheKey
        { public abstract string GetKey();
        } class StringKey : CacheKey
        { string _key; public StringKey(string key)
            {
                _key = key;
            } public override string GetKey()
            { return _key;
            }
        } class ParamsKey : CacheKey
        {
            CacheKey _cacheKey; object[] _args; public ParamsKey(CacheKey cacheKey, params object[] args)
            {
                _cacheKey = cacheKey;
                _args = args;
            } public override string GetKey()
            { var format = _cacheKey.GetKey(); return string.Format(format, _args);
            }
        } public static void RunTest()
        { var key1 = new StringKey("SystemName");
            Console.WriteLine(key1.GetKey());

            key1 = new StringKey("NewsDetails_{0}"); var key2 = new ParamsKey(key1, 23);
            Console.WriteLine(key2.GetKey());
        }
    }

第二个是 Python 里的偏函数概念(其实很简单,就是设置一个函数的部分参数的默认值,生成新的函数),用 C#简单表示一下如下:

    /// <summary>
    /// 通过设定参数的默认值,可以降低函数调用的难度 /// </summary>
    class ThinkPartialFunction
    { static int Multiply(int x, int y)
        { return x * y;
        } static int MultiplyBy2(int x)
        { return Multiply(x, 2);
        } static Func<int, int> BuildMultiplyBy(int y)
        { return (x) => Multiply(x, y);
        } //python  functools.partial(Multiply,y=2)
        static Func<int, int> BuildPartial(Func<int, int, int> fun, int y)
        { return (x) => fun(x, y);
        }
    }

回过头来再看 CacheKey,应该就是装饰器模式的一种变种应用吧。但是设计的时候我可没想到什么装饰器,对设计模式也并不熟识。列出这两点,也是方便大家理解 CacheKey 的设计。

  • .NET
    27 引用 • 6 回帖 • 5 关注
  • 缓存
    42 引用 • 70 回帖

相关帖子

欢迎来到这里!

我们正在构建一个小众社区,大家在这里相互信任,以平等 • 自由 • 奔放的价值观进行分享交流。最终,希望大家能够找到与自己志同道合的伙伴,共同成长。

注册 关于
请输入回帖内容 ...

推荐标签 标签

  • SEO

    发布对别人有帮助的原创内容是最好的 SEO 方式。

    35 引用 • 200 回帖 • 24 关注
  • Electron

    Electron 基于 Chromium 和 Node.js,让你可以使用 HTML、CSS 和 JavaScript 构建应用。它是一个由 GitHub 及众多贡献者组成的活跃社区共同维护的开源项目,兼容 Mac、Windows 和 Linux,它构建的应用可在这三个操作系统上面运行。

    15 引用 • 136 回帖 • 8 关注
  • JavaScript

    JavaScript 一种动态类型、弱类型、基于原型的直译式脚本语言,内置支持类型。它的解释器被称为 JavaScript 引擎,为浏览器的一部分,广泛用于客户端的脚本语言,最早是在 HTML 网页上使用,用来给 HTML 网页增加动态功能。

    710 引用 • 1173 回帖 • 176 关注
  • Linux

    Linux 是一套免费使用和自由传播的类 Unix 操作系统,是一个基于 POSIX 和 Unix 的多用户、多任务、支持多线程和多 CPU 的操作系统。它能运行主要的 Unix 工具软件、应用程序和网络协议,并支持 32 位和 64 位硬件。Linux 继承了 Unix 以网络为核心的设计思想,是一个性能稳定的多用户网络操作系统。

    915 引用 • 931 回帖
  • API

    应用程序编程接口(Application Programming Interface)是一些预先定义的函数,目的是提供应用程序与开发人员基于某软件或硬件得以访问一组例程的能力,而又无需访问源码,或理解内部工作机制的细节。

    76 引用 • 421 回帖
  • ngrok

    ngrok 是一个反向代理,通过在公共的端点和本地运行的 Web 服务器之间建立一个安全的通道。

    7 引用 • 63 回帖 • 596 关注
  • Mac

    Mac 是苹果公司自 1984 年起以“Macintosh”开始开发的个人消费型计算机,如:iMac、Mac mini、Macbook Air、Macbook Pro、Macbook、Mac Pro 等计算机。

    164 引用 • 594 回帖 • 1 关注
  • 笔记

    好记性不如烂笔头。

    303 引用 • 777 回帖
  • 微服务

    微服务架构是一种架构模式,它提倡将单一应用划分成一组小的服务。服务之间互相协调,互相配合,为用户提供最终价值。每个服务运行在独立的进程中。服务于服务之间才用轻量级的通信机制互相沟通。每个服务都围绕着具体业务构建,能够被独立的部署。

    96 引用 • 155 回帖
  • TextBundle

    TextBundle 文件格式旨在应用程序之间交换 Markdown 或 Fountain 之类的纯文本文件时,提供更无缝的用户体验。

    1 引用 • 2 回帖 • 44 关注
  • H2

    H2 是一个开源的嵌入式数据库引擎,采用 Java 语言编写,不受平台的限制,同时 H2 提供了一个十分方便的 web 控制台用于操作和管理数据库内容。H2 还提供兼容模式,可以兼容一些主流的数据库,因此采用 H2 作为开发期的数据库非常方便。

    11 引用 • 54 回帖 • 641 关注
  • SendCloud

    SendCloud 由搜狐武汉研发中心孵化的项目,是致力于为开发者提供高质量的触发邮件服务的云端邮件发送平台,为开发者提供便利的 API 接口来调用服务,让邮件准确迅速到达用户收件箱并获得强大的追踪数据。

    2 引用 • 8 回帖 • 438 关注
  • HBase

    HBase 是一个分布式的、面向列的开源数据库,该技术来源于 Fay Chang 所撰写的 Google 论文 “Bigtable:一个结构化数据的分布式存储系统”。就像 Bigtable 利用了 Google 文件系统所提供的分布式数据存储一样,HBase 在 Hadoop 之上提供了类似于 Bigtable 的能力。

    17 引用 • 6 回帖 • 44 关注
  • ZooKeeper

    ZooKeeper 是一个分布式的,开放源码的分布式应用程序协调服务,是 Google 的 Chubby 一个开源的实现,是 Hadoop 和 HBase 的重要组件。它是一个为分布式应用提供一致性服务的软件,提供的功能包括:配置维护、域名服务、分布式同步、组服务等。

    59 引用 • 29 回帖 • 15 关注
  • MyBatis

    MyBatis 本是 Apache 软件基金会 的一个开源项目 iBatis,2010 年这个项目由 Apache 软件基金会迁移到了 google code,并且改名为 MyBatis ,2013 年 11 月再次迁移到了 GitHub。

    170 引用 • 414 回帖 • 431 关注
  • Telegram

    Telegram 是一个非盈利性、基于云端的即时消息服务。它提供了支持各大操作系统平台的开源的客户端,也提供了很多强大的 APIs 给开发者创建自己的客户端和机器人。

    5 引用 • 35 回帖 • 1 关注
  • Ant-Design

    Ant Design 是服务于企业级产品的设计体系,基于确定和自然的设计价值观上的模块化解决方案,让设计者和开发者专注于更好的用户体验。

    17 引用 • 23 回帖 • 1 关注
  • iOS

    iOS 是由苹果公司开发的移动操作系统,最早于 2007 年 1 月 9 日的 Macworld 大会上公布这个系统,最初是设计给 iPhone 使用的,后来陆续套用到 iPod touch、iPad 以及 Apple TV 等产品上。iOS 与苹果的 Mac OS X 操作系统一样,属于类 Unix 的商业操作系统。

    84 引用 • 139 回帖
  • Netty

    Netty 是一个基于 NIO 的客户端-服务器编程框架,使用 Netty 可以让你快速、简单地开发出一个可维护、高性能的网络应用,例如实现了某种协议的客户、服务端应用。

    49 引用 • 33 回帖 • 22 关注
  • Wide

    Wide 是一款基于 Web 的 Go 语言 IDE。通过浏览器就可以进行 Go 开发,并有代码自动完成、查看表达式、编译反馈、Lint、实时结果输出等功能。

    欢迎访问我们运维的实例: https://wide.b3log.org

    30 引用 • 218 回帖 • 604 关注
  • 锤子科技

    锤子科技(Smartisan)成立于 2012 年 5 月,是一家制造移动互联网终端设备的公司,公司的使命是用完美主义的工匠精神,打造用户体验一流的数码消费类产品(智能手机为主),改善人们的生活质量。

    4 引用 • 31 回帖 • 10 关注
  • Rust

    Rust 是一门赋予每个人构建可靠且高效软件能力的语言。Rust 由 Mozilla 开发,最早发布于 2014 年 9 月。

    57 引用 • 22 回帖
  • Maven

    Maven 是基于项目对象模型(POM)、通过一小段描述信息来管理项目的构建、报告和文档的软件项目管理工具。

    185 引用 • 318 回帖 • 348 关注
  • Ubuntu

    Ubuntu(友帮拓、优般图、乌班图)是一个以桌面应用为主的 Linux 操作系统,其名称来自非洲南部祖鲁语或豪萨语的“ubuntu”一词,意思是“人性”、“我的存在是因为大家的存在”,是非洲传统的一种价值观,类似华人社会的“仁爱”思想。Ubuntu 的目标在于为一般用户提供一个最新的、同时又相当稳定的主要由自由软件构建而成的操作系统。

    123 引用 • 168 回帖
  • App

    App(应用程序,Application 的缩写)一般指手机软件。

    90 引用 • 383 回帖 • 1 关注
  • 思源笔记

    思源笔记是一款隐私优先的个人知识管理系统,支持完全离线使用,同时也支持端到端加密同步。

    融合块、大纲和双向链接,重构你的思维。

    18591 引用 • 69184 回帖 • 1 关注
  • MySQL

    MySQL 是一个关系型数据库管理系统,由瑞典 MySQL AB 公司开发,目前属于 Oracle 公司。MySQL 是最流行的关系型数据库管理系统之一。

    675 引用 • 535 回帖