zoukankan      html  css  js  c++  java
  • webapi 自定义缓存实现

    定义一个Filter

      public class MyOutputCacheAttribute : ActionFilterAttribute
        {
    
            MemoryCacheDefault _cache = new MemoryCacheDefault();
    
    
    
            /// <summary>
            /// 客户端缓存(用户代理)缓存相对过期时间
            /// </summary>
            public int ClientCacheExpiration { set; get; }
    
    
            /// <summary>
            /// 服务端缓存过期时间
            /// </summary>
            public int ServerCacheExpiration { set; get; }
    
    
    
    
            /// <summary>
            /// 
            /// </summary>
            /// <param name="clientCacheExpiration">客户端过期时间。单位为秒,默认为600秒(10分钟)</param>
            /// <param name="cerverCacheExpiration">服务端过期时间。单位为秒,默认为7200秒(120分钟)</param>
            public MyOutputCacheAttribute(int clientCacheExpiration = 600, int serverCacheExpiration = 7200)
            {
                this.ClientCacheExpiration = clientCacheExpiration;
                this.ServerCacheExpiration = serverCacheExpiration;
            }
    
    
    
            /// <summary>
            /// 判定是否用缓存中的内容,还是执行action是去取内容
            /// 注:一旦给actionContext.Response赋值了,则会使用这个值来输出响应
            /// </summary>
            /// <param name="actionContext"></param>
            public override void OnActionExecuting(System.Web.Http.Controllers.HttpActionContext actionContext)
            {
    
                // ****************************************   穿透缓存 *********************************************
                // 若无缓存,则直接返回
                string cacheKey = getCacheKey(actionContext);
                if (_cache.Contains(cacheKey) == false)
                    return;
                if (_cache.Contains(cacheKey + ":etag") == false)
                    return;
    
                // ****************************************   使用缓存  *********************************************
                // 获取缓存中的etag
                var etag = _cache.Get<string>(cacheKey + ":etag");
                // 若etag没有被改变,则返回304,
                if (actionContext.Request.Headers.IfNoneMatch.Any(x => x.Tag == etag))  //IfNoneMatch为空时,返回的是一个集合对象
                {
                    actionContext.Response = actionContext.Request.CreateResponse(HttpStatusCode.NotModified); // 响应对象还没有生成,需要先生成 // 304 not modified
                    actionContext.Response.Headers.CacheControl = GetDefaultCacheControl();
                    actionContext.Response.Headers.ETag = new EntityTagHeaderValue(etag);
                    return;
                }
                else //从缓存中返回响应
                {
                    actionContext.Response = actionContext.Request.CreateResponse(); // 响应对象还没有生成,需要先生成
                    // 设置协商缓存:etag
                    actionContext.Response.Headers.ETag = new EntityTagHeaderValue(etag); //用缓存中的值(为最新的)更新它
                    // 设置user agent的本地缓存
                    actionContext.Response.Headers.CacheControl = GetDefaultCacheControl();
    
                    // 从缓存中取中响应内容
                    var content = _cache.Get<byte[]>(cacheKey);
                    actionContext.Response.Content = new ByteArrayContent(content);
                    return;
                }
            }
    
    
    
    
    
    
            /// <summary>
            ///  将输出保存在缓存中。
            /// </summary>
            /// <param name="actionExecutedContext"></param>
            /// <param name="cancellationToken"></param>
            /// <returns></returns>
            public override async System.Threading.Tasks.Task OnActionExecutedAsync(HttpActionExecutedContext actionExecutedContext, System.Threading.CancellationToken cancellationToken)
            {
                // 响应对象已经生成,可以直接调用   actionExecutedContext.Response
    
                // 设置协商缓存:etag
                EntityTagHeaderValue etag = new EntityTagHeaderValue(""" + Guid.NewGuid().ToString() + """);  // 根据http协议, ETag的值必须用引号包含起来
                actionExecutedContext.Response.Headers.ETag = etag;
                // 设置user agent的本地缓存
                actionExecutedContext.Response.Headers.CacheControl = GetDefaultCacheControl();
    
                // actionExecutedContext.Response.Headers.Remove("Content-Length"); // 改变了值,它会发生变化。删除它的话,后续的程序会自动地再计算
    
                // 保存到缓存
                string cacheKey = getCacheKey(actionExecutedContext.ActionContext);
                var contentBytes = await actionExecutedContext.Response.Content.ReadAsByteArrayAsync();
    
                _cache.Add(cacheKey + ":etag", etag.Tag, DateTimeOffset.Now.AddSeconds(this.ServerCacheExpiration));
                _cache.Add(cacheKey, contentBytes, DateTimeOffset.Now.AddSeconds(this.ServerCacheExpiration));
    
            }
    
    
    
    
    
            /// <summary>
            /// 默认的用户代理本地缓存配置,10分钟的相对过期时间
            /// 对应响应header中的 Cache-Control
            /// 这里设置里面的子项max-age。如:Cache-Control: max-age=3600
            /// </summary>
            /// <returns></returns>
            CacheControlHeaderValue GetDefaultCacheControl()
            {
                CacheControlHeaderValue control = new CacheControlHeaderValue();
                control.MaxAge = TimeSpan.FromSeconds(this.ClientCacheExpiration);  // 它对应响应头中的 cacheControl :max-age=10项
    
                return control;
            }
    
    
    
    
            /// <summary>
            /// 
            /// </summary>
            /// <param name="actionContext"></param>
            /// <returns></returns>
            string getCacheKey(System.Web.Http.Controllers.HttpActionContext actionContext)
            {
                string cacheKey = null;
    
                cacheKey = actionContext.Request.RequestUri.PathAndQuery; // 路径和查询部分,如: /api/values/1?ee=33&oo=5
    
                // 对路径中的参数进行重排,升序排列
    
                //string controllerName = actionContext.ControllerContext.ControllerDescriptor.ControllerName;
                //string actionName = actionContext.ActionDescriptor.ActionName;
    
                //if (actionContext.ActionArguments.ContainsKey("id") == false)
                //{
                //    cacheKey = controllerName + "/" + actionName;
                //}
                //else
                //{
                //    string id = actionContext.ActionArguments["id"].ToString();
                //    cacheKey = controllerName + "/" + actionName + "/" + id;
                //}
    
                return cacheKey;
            }
    
    
        }

     上面的这段代码严格遵循RFC2626中定义的缓存协议。

    定义一个服务器端缓存实现

    这里采用MemoryCache,也可以采用memcached, redis之类的。

    public class MemoryCacheDefault
    {
        private static readonly MemoryCache _cache = MemoryCache.Default;
    
        public void RemoveStartsWith(string key)
        {
            lock (_cache)
            {
                _cache.Remove(key);
            }
        }
    
        public T Get<T>(string key) where T : class
        {
            var o = _cache.Get(key) as T;
            return o;
        }
    
    
    
        [Obsolete("Use Get<T> instead")]
        public object Get(string key)
        {
            return _cache.Get(key);
        }
    
    
    
    
    
        public void Remove(string key)
        {
            lock (_cache)
            {
                _cache.Remove(key);
            }
        }
    
        public bool Contains(string key)
        {
            return _cache.Contains(key);
        }
    
        public void Add(string key, object o, DateTimeOffset expiration, string dependsOnKey = null)
        {
            var cachePolicy = new CacheItemPolicy
            {
                AbsoluteExpiration = expiration
            };
    
            if (!string.IsNullOrWhiteSpace(dependsOnKey))
            {
                cachePolicy.ChangeMonitors.Add(
                    _cache.CreateCacheEntryChangeMonitor(new[] { dependsOnKey })
                );
            }
            lock (_cache)
            {
                _cache.Add(key, o, cachePolicy);
            }
        }
    
    
        public IEnumerable<string> AllKeys
        {
            get
            {
                return _cache.Select(x => x.Key);
            }
        }
    }

    将filter应用到action中

        public class ValuesController : ApiController
        {
    
            [MyOutputCache(10)]
            public string Get(int id)
            {
                return "value" + id.ToString();
            }
        }
  • 相关阅读:
    Goahead在linux环境下安装部署
    vim卡住怎么办
    Clickhouse 实现 row number功能
    JavaScript ES6 模块化
    MySQL012事务的四个基本特征是什么
    MySQL015简述mysql中索引类型有哪些,以及对数据库的性能的影响
    MySQL010MySQL执行计划怎么看
    JavaScript ES6 Promise
    MySQL011如何处理MySQL的慢查询
    MySQL009MySQL为什么需要主从复制和读写分离
  • 原文地址:https://www.cnblogs.com/dehai/p/5068911.html
Copyright © 2011-2022 走看看