在web应用程序中使用MemcachedClient
本文来自:http://www.cnblogs.com/yukaizhao/archive/2008/11/10/memcached_client_usage.html
一. 背景:
在大访问量的web程序开发中,数据库常常会称为性能的瓶颈。为了缓解数据库的压力,我们频繁的使用缓存,而asp.net自带的Cache很强大,但是有先天的不足,它是进程内的缓存,当站点由多台服务器负载均衡时,当缓存在有数据更新时,我们不能同时将更新后的数据同步到两台或多台web server上。所幸的是老外的大牛开发了memcached分布式缓存,它的性能非凡,memcached常用的.net的client类库有两个分别是:http://code.google.com/p/beitmemcached/,http://sourceforge.net/projects/memcacheddotnet/,我个人推荐使用后一个。
有关memcached的介绍和在控制台中的简单调用请参考:分布式缓存系统Memcached简介与实践
二. 如何使用
- 安装memcached server端,下载dot net版本的客户端,请参考分布式缓存系统Memcached简介与实践
- 在asp.net项目中使用memcached客户端访问服务端
思路:
1) 我们在站点启动时即启动memcached的client端,在站点停止时停掉client端,释放占用的端口资源;
这一点只要把启动和关闭放到Global的Application_Start和Application_End中执行,即可。由于整个网站的运行期间,我们都希望不需要再重新生成MemcachedClient类的实例,所以我们在Global中声明了一个静态的实例来存放此Client。并用一个public的属性RemoteCache和IsRemoteCacheAvailable来暴露MemcachedClient的引用和其是否可用。
2) 为了方便使用我们需要建一个System.Web.UI.Page类的基类PageBase,在此基类中引用Memcached client的客户端访问类实例,然后新建的所有aspx页面的基类继承PageBase就可以了。
这儿也很容已做到,我们只需找到 Global appInstance = HttpContext.Current.ApplicationInstance as Global;然后引用Global中暴露的两个属性即可。需要注意的是在程序中引用Global是通过HttpContext.Current.ApplicationInstance而不是HttpContext.Current.Applicatioin,后者只是一个键值对的集合,而前者是网站的HttpApplication的实例
请参考Global和PageBase的代码:
- using System;
- using System.Data;
- using System.Configuration;
- using System.Collections;
- using System.Web;
- using System.Web.Security;
- using System.Web.SessionState;
- using Memcached.ClientLibrary;
- using log4net;
- namespace Forum.UI
- {
- public class Global : System.Web.HttpApplication
- {
- protected static ILog AppLog = LogManager.GetLogger(typeof(Global));
- public override void Init()
- {
- this.Error += new EventHandler(Application_Error);
- base.Init();
- }
- protected void Application_Error(object sender, EventArgs e)
- {
- #if !DEBUG
- Exception ex = Server.GetLastError();
- AppLog.Error(ex.Message, ex);
- Server.ClearError();
- Response.Redirect("~/error/500.htm", true);
- #endif
- }
- protected void Application_Start(object sender, EventArgs e)
- {
- SetupMemcachedClient();
- }
- protected void Application_End(object sender, EventArgs e)
- {
- ShutdownMemecachedClient();
- }
- #region RemoteCache
- private static MemcachedClient mc = null;
- private const string MEMCACHED_INSTANCE_NAME = "Memcached-Forum";
- /// <summary>
- /// 返回MemcachedClient是否可用
- /// </summary>
- public bool IsRemoteCacheAvailable
- {
- get
- {
- return mc != null;
- }
- }
- /// <summary>
- /// 外部访问入口
- /// </summary>
- public MemcachedClient RemoteCache
- {
- get
- {
- return mc;
- }
- }
- /// <summary>
- /// 关闭占用的tcp端口资源
- /// </summary>
- private void ShutdownMemecachedClient()
- {
- SockIOPool pool = SockIOPool.GetInstance(MEMCACHED_INSTANCE_NAME);
- if (pool != null) pool.Shutdown();
- }
- /// <summary>
- /// 启动memcachedClient,给client赋予指定参数
- /// </summary>
- private void SetupMemcachedClient()
- {
- string memcachedServers = ConfigurationManager.AppSettings["memcachedServers"];
- string[] servers = memcachedServers.Split(';');
- SockIOPool pool = SockIOPool.GetInstance(MEMCACHED_INSTANCE_NAME);
- pool.SetServers(servers);
- pool.InitConnections = 3;
- pool.MinConnections = 3;
- pool.MaxConnections = 5;
- pool.SocketConnectTimeout = 1000;
- pool.SocketTimeout = 3000;
- pool.MaintenanceSleep = 30;
- pool.Failover = true;
- pool.Nagle = false;
- pool.Initialize();
- mc = new MemcachedClient();
- mc.PoolName = MEMCACHED_INSTANCE_NAME;
- mc.EnableCompression = false;
- }
- #endregion
- }
- }
- using System;
- using System.Data;
- using System.Configuration;
- using System.Web;
- using System.Web.Security;
- using System.Web.UI;
- using System.Web.UI.WebControls;
- using System.Web.UI.WebControls.WebParts;
- using System.Web.UI.HtmlControls;
- using System.Web.Caching;
- using Memcached.ClientLibrary;
- using System.Collections.Generic;
- using log4net;
- using Forum.Models;
- using Forum.UI.Proxy;
- namespace Forum.UI
- {
- public class PageBase : Page
- {
- #region RemoteCache & IsRemoteCacheAvailable
- protected MemcachedClient RemoteCache
- {
- get
- {
- if (HttpContext.Current.ApplicationInstance == null) return null;
- Global appInstance = HttpContext.Current.ApplicationInstance as Global;
- if (appInstance == null) return null;
- return appInstance.RemoteCache;
- }
- }
- protected bool IsRemoteCacheAvailable
- {
- get
- {
- return RemoteCache != null;
- }
- }
- #endregion
- }
- }
三. 后记
仅仅是个人使用心得,如果有问题,请回复讨论。
在web应用程序中使用MemcachedClient的更多相关文章
- 在 ASP.NET MVC Web 应用程序中输出 RSS Feeds
RSS全称Really Simple Syndication.一些更新频率较高的网站可以通过RSS让订阅者快速获取更新信息.RSS文档需遵守XML规范的,其中必需包含标题.链接.描述信息,还可以包含发 ...
- Java嵌入式数据库H2学习总结(二)——在Web应用程序中使用H2数据库
一.搭建测试环境和项目 1.1.搭建JavaWeb测试项目 创建一个[H2DBTest]JavaWeb项目,找到H2数据库的jar文件,如下图所示: H2数据库就一个jar文件,这个Jar文件里面包含 ...
- 在Web应用程序中执行计划任务(多线程)
在业务复杂的应用程序中,有时候会要求一个或者多个任务在一定的时间或者一定的时间间隔内计划进行,比如定时备份或同步数据库,定时发送电子邮件等,我们称之为计划任务.实现计划任务的方法也有很多,可以采用SQ ...
- 5.把报表集成到Web应用程序中-生成网页和导出两种方式
转自:https://wenku.baidu.com/view/104156f9770bf78a65295462.html 第四部分,把报表集成到Web应用程序中 用MyEclipse新建一个Web ...
- Asp.NetCore Web应用程序中的请求管道和中间件
你是否会迷惑当我们请求一个ASP.NetWeb应用程序以后,它是怎么处理这些请求的,后台是怎么工作的,今天就讲一下Asp.NetCore Web应用程序中的请求处理过程. 上一节,我们讲到,Start ...
- Web 应用程序中的安全向量 – ASP.NET MVC 4 系列
Web 程序运行在标准的.基于文本的协议(HTTP 和 HTML)之上,所以特别容易受到自动攻击的伤害.本章主要介绍黑客如何滥用应用程序,以及针对这些问题的应对措施. 威胁:跨站脚本 ...
- 7、Web应用程序中的安全向量 -- 使用Retail部署配置
该方法不需要胡乱地编辑各个配置设置,而是利用了ASP.NET特性:Retail部署配置. 部署配置是服务器的machine.config文件(在%windir%\Microsoft.NET\Frame ...
- 6、Web应用程序中的安全向量 -- customErrors(适当的错误报告和堆栈跟踪)
几乎所有的网站在开发过程中都在web.config文件中设置了特性<customErrors mode="off">. customErrors模式有3个可选的设置项: ...
- 3、Web应用程序中的安全向量 -- cookie盗窃
作为用户,为了防止cookie盗窃,可以在浏览器设置中选择"禁用cookie",但是这样做很可能导致在访问某个站点的时候弹出警告"该站点必须使用cookie". ...
随机推荐
- 编码规范(CSS)
code { font-family: "PT Mono", Menlo, "Courier New", monospace; padding: 2px 4px ...
- SAX解析和生成XML文档
原创作品,允许转载,转载时请务必以超链接形式标明文章 原始出处 .作者信息和本人声明.否则将追究法律责任. 作者: 永恒の_☆ 地址: http://blog.csdn.net/chenghui031 ...
- 配置元素customErrors
Asp.net配置文件的配置方式,其实在MSDN里面是写得最清楚的了.可惜之前一直未曾了解到MSDN的强大. 先贴个地址:http://msdn.microsoft.com/zh-cn/library ...
- iOS 开发~UIWindow
比如下面例子: 写出第一个iOS的程序,在界面上显示“Hello World” 1.如何新建一个工程 iOS—>Single View Application—>工程名.保存位置 2.运行 ...
- enumerateObjectsUsingBlock VS for(... in ...)
enumerateObjectsUsingBlock VS for(... in ...) for(... in ...)用起来非常方便.简洁,同时enumerateObjectsUsingBlock ...
- 不能修改“System Roots”钥匙串 即下载的.cer 文件添加不到钥匙串
双击提示 :不能修改“System Roots”钥匙串要更改根证书是否会被信任,请在“钥匙串访问”中打开它,然后修改它的信任设置. 解决办法:添加到 登录或显示LOGIN的 keychain(记 ...
- c#窗体的传值方法
了解了窗体的显示相关知识,接着总结一下窗体的传值方法: .通过构造函数 特点:传值是单向的(不可以互相传值),实现简单 实现代码如下: 在窗体Form2中 int value1; ...
- BestCoder Round #20 部分题解(A,B,C)(hdu5123,5124,5125)
转载请注明出处: http://www.cnblogs.com/fraud/ ——by fraud who is the best? Time Limit: 2000/1000 MS ...
- Eclipse控制台显示Tomcat日志
今天看一篇学习Struts的博文,文章里面提到从生成的日志,结果,怎么鼓捣都看不到.心情也跟着烦躁了.于是晚饭后出去散步,冷静一下,然后决定晚上一定搞掂这个问题.这不,搞掂了,写篇博文记录一下. St ...
- [C++程序设计]函数的递归调用
在调用一个函数的过程中又出现直接或间接地调用 该函数本身,称为函数的递归(recursive)调用. 包含递归调用的函数称为递归函数. 在实现递归时,在时间和空间上的开销比较大 求n! #includ ...