热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

无缝的缓存读取:双存储缓存策略

起最近在做一个WEB的数据统计的优化,但是由于数据量大,执行一次SQL统计要比较长的时间(一般700ms算是正常)。正常的做

最近在做一个WEB的数据统计的优化,但是由于数据量大,执行一次SQL统计要比较长的时间(一般700ms算是正常)。

正常的做法只要加个缓存就好了。

但是同时业务要求此数据最多1分钟就要更新,而且这一分种内数据可能会有较多变化(而且原系统不太易扩展)。

也就是说缓存1分钟就要失效重新统计,而且用户访问这页还很是频繁,如果使用一般缓存那么用户体验很差而且很容易造成超时。

 

看到以上需求,第一个进入我大脑的就是从前做游戏时接触到的DDraw的双缓冲显示方式。

image

在第一帧显示的同时,正在计算第二帧,这样读取和计算就可以分开了,也就避免了读取时计算,提高了用户体验。

我想当然我们也可以将这种方式用于缓存的策略中,但这样用空间换取时间的方式还是得权衡的,因为并不是所有时候都值得这么做,但这里我觉得这样做应该是最好的方式了。

注:为了可以好好演示,本篇中的缓存都以IEnumerable的形式来存储,当然这个文中原理也可以应用在WebCache中。

这里我使用以下数据结构做为存储单元:

namespace CHCache {///

/// 缓存介质/// public class Medium {/// /// 主要存储介质/// public object Primary { get; set; }/// /// 次要存储介质/// public object Secondary { get; set; }/// /// 是否正在使用主要存储/// public bool IsPrimary { get; set; }/// /// 是否正在更新/// public bool IsUpdating { get; set; }/// /// 是否更新完成/// public bool IsUpdated { get; set; }}
}

有了这个数据结构我们就可以将数据实现两份存储。再利用一些读写策略就可以实现上面我们讲的缓存方式。

整个的缓存我们使用如下缓存类来控制:

/** http://www.cnblogs.com/chsword/* chsword* Date: 2009-3-31* Time: 17:00* */
using System;
using System.Collections;
using System.Collections.Generic;
using System.Threading;
namespace CHCache {///

/// 双存储的类/// public class DictionaryCache : IEnumerable {/// /// 在此缓存构造时初始化字典对象/// public DictionaryCache(){Store &#61; new Dictionary<string, Medium>();}public void Add(string key,Func<object> func){if (Store.ContainsKey(key)) {//修改&#xff0c;如果已经存在&#xff0c;再次添加时则采用其它线程var elem &#61; Store[key];if (elem.IsUpdating)return; //正在写入未命中var th &#61; new ThreadHelper(elem, func);//ThreadHelper将在下文提及,是向其它线程传参用的var td &#61; new Thread(th.Doit);td.Start();}else {//首次添加时可能也要读取&#xff0c;所以要本线程执行Console.WriteLine("Begin first write");Store.Add(key, new Medium {IsPrimary &#61; true, Primary &#61; func()});Console.WriteLine("End first write");}}/// /// 读取时所用的索引/// /// /// public object this[string key] {get {if (!Store.ContainsKey(key))return null;var elem &#61; Store[key];if (elem.IsUpdated) {//如果其它线程更新完毕&#xff0c;则将主次转置elem.IsUpdated &#61; false;elem.IsPrimary &#61; !elem.IsPrimary;} var ret &#61; elem.IsPrimary ? elem.Primary : elem.Secondary;var b &#61; elem.IsPrimary ? " from 1" : " form 2";return ret &#43; b;}}Dictionary<string, Medium> Store { get; set; }public IEnumerator GetEnumerator() {return ((IEnumerable)Store).GetEnumerator();}}
}

这里我只实现了插入一个缓存&#xff0c;以及读取的方法。

我读取缓存单元的逻辑是这样的

image 

从2个不同缓存读取当然是很容易了&#xff0c;但是比较复杂的就是向缓存写入的过程&#xff1a;

image

这里读取数据以及写入缓存时我使用了一个委托&#xff0c;在其它线程中仅在需要执行时才会执行。

这里除了首次写入缓存占用主线程时间&#xff08;读取要等待&#xff09;以外&#xff0c;其它时间都可以无延时的读取&#xff0c;实现了无缝的缓存。

但我们在委托中要操作缓存的元素Medium,所以要传递参数进其它线程&#xff0c;所以我这里使用了一个辅助类来传递参数进入其它线程&#xff1a;

using System;
namespace CHCache {///

/// 一个线程Helper&#xff0c;用于帮助多抛出线程时传递参数/// public class ThreadHelper {Func<object> Fun { get; set; }Medium Medium { get; set; }/// /// 通过构造函数来传递参数/// /// 缓存单元/// 读取数据的委托public ThreadHelper(Medium m,Func<object> fun) {Medium &#61; m;Fun &#61; fun;}/// /// 线程入口&#xff0c;ThreadStart委托所对应的方法/// public void Doit(){Medium.IsUpdating &#61; true;if (Medium.IsPrimary) {Console.WriteLine("Begin write to 2.");var ret &#61; Fun.Invoke();Medium.Secondary &#61; ret;Console.WriteLine("End write to 2.");}else {Console.WriteLine("Begin write to 1.");var ret &#61; Fun.Invoke();Medium.Primary &#61; ret;Console.WriteLine("End write to 1.");}Medium.IsUpdated &#61; true;Medium.IsUpdating &#61; false;}}
}

这样我们就实现了在另个线程读取数据的过程&#xff0c;这样就在任何时候读取数据时都会无延时直接读取了。

最后我们写一个主函数来测试一下效果

/** http://www.cnblogs.com/chsword/* chsword* Date: 2009-3-31* Time: 16:53*/
using System;
using System.Threading;
namespace CHCache
{class Program{public static void Main(string[] args){var cache &#61; new DictionaryCache();Console.WriteLine("Init...4s&#xff0c;you can press the CTRL&#43;C to close the console window.");while (true){cache.Add("1", GetValue);Thread.Sleep(1000);Console.WriteLine(cache["1"]);}}///

/// 获取数据的方法&#xff0c;假设是从数据库读取的&#xff0c;费时约4秒/// /// static object GetValue(){Thread.Sleep(4000);return DateTime.Now;}}
}

得到如下数据&#xff1a;

image

这样就实现了平滑的读取缓存数据而没有任何等待时间

当然这里还有些问题&#xff0c;比如说传递不同参数时的解决方法&#xff0c;但是由于我仅是在一个统计时需要这种缓存提高性能&#xff0c;所以暂没有考虑通用的传参方式。

如果大家对这个话题感兴趣&#xff0c;欢迎讨论。

 

示例下载&#xff1a;

 

Cat Chen一语提醒&#xff0c;其实做缓存的提前加载没有必要使用2个缓存的&#xff0c;于是将列子改了改&#xff1a;无缝缓存读取简化&#xff1a;仅Lambda表达式传递委托


转载于:https://www.cnblogs.com/chsword/archive/2009/04/01/ch2cache.html


推荐阅读
  • 本文介绍了一个在线急等问题解决方法,即如何统计数据库中某个字段下的所有数据,并将结果显示在文本框里。作者提到了自己是一个菜鸟,希望能够得到帮助。作者使用的是ACCESS数据库,并且给出了一个例子,希望得到的结果是560。作者还提到自己已经尝试了使用"select sum(字段2) from 表名"的语句,得到的结果是650,但不知道如何得到560。希望能够得到解决方案。 ... [详细]
  • 在重复造轮子的情况下用ProxyServlet反向代理来减少工作量
    像不少公司内部不同团队都会自己研发自己工具产品,当各个产品逐渐成熟,到达了一定的发展瓶颈,同时每个产品都有着自己的入口,用户 ... [详细]
  • 本文详细介绍了Spring的JdbcTemplate的使用方法,包括执行存储过程、存储函数的call()方法,执行任何SQL语句的execute()方法,单个更新和批量更新的update()和batchUpdate()方法,以及单查和列表查询的query()和queryForXXX()方法。提供了经过测试的API供使用。 ... [详细]
  • 前景:当UI一个查询条件为多项选择,或录入多个条件的时候,比如查询所有名称里面包含以下动态条件,需要模糊查询里面每一项时比如是这样一个数组条件:newstring[]{兴业银行, ... [详细]
  • 本文讨论了Kotlin中扩展函数的一些惯用用法以及其合理性。作者认为在某些情况下,定义扩展函数没有意义,但官方的编码约定支持这种方式。文章还介绍了在类之外定义扩展函数的具体用法,并讨论了避免使用扩展函数的边缘情况。作者提出了对于扩展函数的合理性的质疑,并给出了自己的反驳。最后,文章强调了在编写Kotlin代码时可以自由地使用扩展函数的重要性。 ... [详细]
  • 在Android开发中,使用Picasso库可以实现对网络图片的等比例缩放。本文介绍了使用Picasso库进行图片缩放的方法,并提供了具体的代码实现。通过获取图片的宽高,计算目标宽度和高度,并创建新图实现等比例缩放。 ... [详细]
  • Spring源码解密之默认标签的解析方式分析
    本文分析了Spring源码解密中默认标签的解析方式。通过对命名空间的判断,区分默认命名空间和自定义命名空间,并采用不同的解析方式。其中,bean标签的解析最为复杂和重要。 ... [详细]
  • 开发笔记:加密&json&StringIO模块&BytesIO模块
    篇首语:本文由编程笔记#小编为大家整理,主要介绍了加密&json&StringIO模块&BytesIO模块相关的知识,希望对你有一定的参考价值。一、加密加密 ... [详细]
  • 本文主要解析了Open judge C16H问题中涉及到的Magical Balls的快速幂和逆元算法,并给出了问题的解析和解决方法。详细介绍了问题的背景和规则,并给出了相应的算法解析和实现步骤。通过本文的解析,读者可以更好地理解和解决Open judge C16H问题中的Magical Balls部分。 ... [详细]
  • 在说Hibernate映射前,我们先来了解下对象关系映射ORM。ORM的实现思想就是将关系数据库中表的数据映射成对象,以对象的形式展现。这样开发人员就可以把对数据库的操作转化为对 ... [详细]
  • http:my.oschina.netleejun2005blog136820刚看到群里又有同学在说HTTP协议下的Get请求参数长度是有大小限制的,最大不能超过XX ... [详细]
  • [大整数乘法] java代码实现
    本文介绍了使用java代码实现大整数乘法的过程,同时也涉及到大整数加法和大整数减法的计算方法。通过分治算法来提高计算效率,并对算法的时间复杂度进行了研究。详细代码实现请参考文章链接。 ... [详细]
  • 本文介绍了Linux Shell中括号和整数扩展的使用方法,包括命令组、命令替换、初始化数组以及算术表达式和逻辑判断的相关内容。括号中的命令将会在新开的子shell中顺序执行,括号中的变量不能被脚本余下的部分使用。命令替换可以用于将命令的标准输出作为另一个命令的输入。括号中的运算符和表达式符合C语言运算规则,可以用在整数扩展中进行算术计算和逻辑判断。 ... [详细]
  • 摘要: 在测试数据中,生成中文姓名是一个常见的需求。本文介绍了使用C#编写的随机生成中文姓名的方法,并分享了相关代码。作者欢迎读者提出意见和建议。 ... [详细]
  • 本文详细介绍了如何使用MySQL来显示SQL语句的执行时间,并通过MySQL Query Profiler获取CPU和内存使用量以及系统锁和表锁的时间。同时介绍了效能分析的三种方法:瓶颈分析、工作负载分析和基于比率的分析。 ... [详细]
author-avatar
月光下大手拉S小手
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有