用Redis实现Session功能
0.什么是Redis
Redis是一个开源的使用ANSI C语言编写、支持网络、可基于内存亦可持久化的日志型、Key-Value数据库,并提供多种语言的API
---维基百科
1.与其他用户状态保存方案比较
一般开发中用户状态使用session或者cookie,两种方式各种利弊。
Session:在InProc模式下容易丢失,并且引起并发问题。如果使用SQLServer或者SQLServer模式又消耗了性能
Cookie则容易将一些用户信息暴露,加解密同样也消耗了性能。
Redis采用这样的方案解决了几个问题,
1.Redis存取速度快。
2.用户数据不容易丢失。
3.用户多的情况下容易支持集群。
4.能够查看在线用户。
5.能够实现用户一处登录。(通过代码实现,后续介绍)
6.支持持久化。(当然可能没什么用)
2.实现思路
1.我们知道session其实是在cookie中保存了一个sessionid,用户每次访问都将sessionid发给服务器,服务器通过ID查找用户对应的状态数据。
在这里我的处理方式也是在cookie中定义一个sessionid,程序需要取得用户状态时将sessionid做为key在Redis中查找。
2.同时session支持用户在一定时间不访问将session回收。
借用Redis中Keys支持过期时间的特性支持这个功能,但是在续期方面需要程序自行拦截请求调用这个方法(demo有例子)
下面开始代码说明
3.Redis调用接口
首先引用ServiceStack相关DLL。
在web.config添加配置,这个配置用来设置Redis调用地址每台服务用【,】隔开。主机写在第一位
- 1 <appSettings>
- 2
- 3 <!--每台Redis之间用,分割.第一个必须为主机-->
- 4 <add key="SessionRedis" value="127.0.0.1:6384,127.0.0.1:6384"/>
- 5
- 6 </appSettings>
初始化配置
- static Managers()
- {
- string sessionRedis= ConfigurationManager.AppSettings["SessionRedis"];
- string timeOut = ConfigurationManager.AppSettings["SessionRedisTimeOut"];
- if (string.IsNullOrEmpty(sessionRedis))
- {
- throw new Exception("web.config 缺少配置SessionRedis,每台Redis之间用,分割.第一个必须为主机");
- }
- if (string.IsNullOrEmpty(timeOut)==false)
- {
- TimeOut = Convert.ToInt32(timeOut);
- }
- var host = sessionRedis.Split(char.Parse(","));
- var writeHost = new string[] { host[0] };
- var readHosts = host.Skip(1).ToArray();
- ClientManagers = new PooledRedisClientManager(writeHost, readHosts, new RedisClientManagerConfig
- {
- MaxWritePoolSize = writeReadCount,//“写”链接池链接数
- MaxReadPoolSize = writeReadCount,//“读”链接池链接数
- AutoStart = true
- });
- }
为了控制方便写了一个委托
- /// <summary>
- /// 写入
- /// </summary>
- /// <typeparam name="F"></typeparam>
- /// <param name="doWrite"></param>
- /// <returns></returns>
- public F TryRedisWrite<F>(Func<IRedisClient, F> doWrite)
- {
- PooledRedisClientManager prcm = new Managers().GetClientManagers();
- IRedisClient client = null;
- try
- {
- using (client = prcm.GetClient())
- {
- return doWrite(client);
- }
- }
- catch (RedisException)
- {
- throw new Exception("Redis写入异常.Host:" + client.Host + ",Port:" + client.Port);
- }
- finally
- {
- if (client != null)
- {
- client.Dispose();
- }
- }
- }
一个调用的例子其他的具体看源码
- /// <summary>
- /// 以Key/Value的形式存储对象到缓存中
- /// </summary>
- /// <typeparam name="T">对象类别</typeparam>
- /// <param name="value">要写入的集合</param>
- public void KSet(Dictionary<string, T> value)
- {
- Func<IRedisClient, bool> fun = (IRedisClient client) =>
- {
- client.SetAll<T>(value);
- return true;
- };
- TryRedisWrite(fun);
- }
4.实现Session
按上面说的给cookie写一个sessionid
- /// <summary>
- /// 用户状态管理
- /// </summary>
- public class Session
- {
- /// <summary>
- /// 初始化
- /// </summary>
- /// <param name="_context"></param>
- public Session(HttpContextBase _context)
- {
- var context = _context;
- var cookie = context.Request.Cookies.Get(SessionName);
- if (cookie == null || string.IsNullOrEmpty(cookie.Value))
- {
- SessionId = NewGuid();
- context.Response.Cookies.Add(new HttpCookie(SessionName, SessionId));
- context.Request.Cookies.Add(new HttpCookie(SessionName, SessionId));
- }
- else
- {
- SessionId = cookie.Value;
- }
- }
- }
去存取用户的方法
- /// <summary>
- /// 获取当前用户信息
- /// </summary>
- /// <typeparam name="T"></typeparam>
- /// <returns></returns>
- public object Get<T>() where T:class,new()
- {
- return new RedisClient<T>().KGet(SessionId);
- }
- /// <summary>
- /// 用户是否在线
- /// </summary>
- /// <returns></returns>
- public bool IsLogin()
- {
- return new RedisClient<object>().KIsExist(SessionId);
- }
- /// <summary>
- /// 登录
- /// </summary>
- /// <typeparam name="T"></typeparam>
- /// <param name="obj"></param>
- public void Login<T>(T obj) where T : class,new()
- {
- new RedisClient<T>().KSet(SessionId, obj, new TimeSpan(0, Managers.TimeOut, 0));
- }
6.续期
默认用户没访问超过30分钟注销用户的登录状态,所以用户每次访问都要将用户的注销时间推迟30分钟
这需要调用Redis的续期方法
- /// <summary>
- /// 延期
- /// </summary>
- /// <param name="key"></param>
- /// <param name="expiresTime"></param>
- public void KSetEntryIn(string key, TimeSpan expiresTime)
- {
- Func<IRedisClient, bool> fun = (IRedisClient client) =>
- {
- client.ExpireEntryIn(key, expiresTime);
- return false;
- };
- TryRedisWrite(fun);
- }
- 封装以后
- /// <summary>
- /// 续期
- /// </summary>
- public void Postpone()
- {
- new RedisClient<object>().KSetEntryIn(SessionId, new TimeSpan(0, Managers.TimeOut, 0));
- }
- 这里我利用了MVC3中的ActionFilter,拦截用户的所有请求
- namespace Test
- {
- public class SessionFilterAttribute : ActionFilterAttribute
- {
- /// <summary>
- /// 每次请求都续期
- /// </summary>
- /// <param name="filterContext"></param>
- public override void OnActionExecuting(ActionExecutingContext filterContext)
- {
- new Session(filterContext.HttpContext).Postpone();
- }
- }
- }
- 在Global.asax中要注册一下
- public static void RegisterGlobalFilters(GlobalFilterCollection filters)
- {
- filters.Add(new SessionFilterAttribute());
- }
- protected void Application_Start()
- {
- RegisterGlobalFilters(GlobalFilters.Filters);
- }
5.调用方式
为了方便调用借用4.0中的新特性,把Controller添加一个扩展属性
- public static class ExtSessions
- {public static Session SessionExt(this Controller controller)
- {
- return new Session(controller.HttpContext);
- }
- }
调用方法
- public class HomeController : Controller
- {
- public ActionResult Index()
- {
- this.SessionExt().IsLogin();
- return View();
- }
- }
6.代码下载
7.后续
SessionManager包含 获取用户列表数量,注销某个用户,根据用户ID获取用户信息,在线用户对象列表,在线用户SessionId列表等方法
后续将实现用户一处登录功能
<script type="text/javascript"><!-- google_ad_client = "ca-pub-1944176156128447"; /* cnblogs 首页横幅 */ google_ad_slot = "5419468456"; google_ad_width = 728; google_ad_height = 90; //--></script><script type="text/javascript" src="http://pagead2.googlesyndication.com/pagead/show_ads.js"></script>
用Redis实现Session功能的更多相关文章
- 分布式中使用Redis实现Session共享(二)
上一篇介绍了一些redis的安装及使用步骤,本篇开始将介绍redis的实际应用场景,先从最常见的session开始,刚好也重新学习一遍session的实现原理.在阅读之前假设你已经会使用nginx+i ...
- Redis+Django(Session,Cookie)的用户系统
一.Django authentication django authentication提供了一个便利的user api接口,无论在py中 request.user,参见Request and re ...
- Redis+Django(Session,Cookie、Cache)的用户系统
转自 http://www.cnblogs.com/BeginMan/p/3890761.html 一.Django authentication django authentication 提供了一 ...
- [转]分布式中使用Redis实现Session共享(二)
本文转自:http://www.cnblogs.com/yanweidie/p/4678095.html 上一篇介绍了一些redis的安装及使用步骤,本篇开始将介绍redis的实际应用场景,先从最常见 ...
- (转)分布式中使用Redis实现Session共享(二)
上一篇介绍了一些redis的安装及使用步骤,本篇开始将介绍redis的实际应用场景,先从最常见的session开始,刚好也重新学习一遍session的实现原理.在阅读之前假设你已经会使用nginx+i ...
- Tomcat7基于Redis的Session共享实战二
目前,为了使web能适应大规模的访问,需要实现应用的集群部署.集群最有效的方案就是负载均衡,而实现负载均衡用户每一个请求都有可能被分配到不固定的服务器上,这样我们首先要解决session的统一来保证无 ...
- 【荐】PHP Session和Cookie,Session阻塞,Session垃圾回收,Redis共享Session,不推荐Memcached保存Session
什么是 Session 在 web 应用开发中,Session 被称为会话.主要被用于保存某个访问者的数据. 由于 HTTP 无状态的特点,服务端是不会记住客户端的,对服务端来说,每一个请求都是全新的 ...
- .Net分布式架构(二):基于Redis的Session共享
一:Session简介 Session是什么呢?简单来说就是服务器给客户端的一个编号.当一台web服务器运行时,可能有若干个用户浏览正在运正在这台服务器上的网站.当每个用户首次与这台web服务器建立连 ...
- linux下实现redis共享session的tomcat集群
为了实现主域名与子域名的下不同的产品间一次登录,到处访问的效果,因此采用rediss实现tomcat的集群效果.基于redis能够异步讲缓存内容固化到磁盘上,从而当服务器意外重启后,仍然能够让sess ...
随机推荐
- python3.5.2本地环境搭建
OS:win7 Download URL:https://www.python.org/downloads/release/python-352/ install 下载二进制安装文件之后,点击安装,一 ...
- 使用xib封装一个view的步骤
1.新建一个xib文件描述一个view的内部结构(假设叫做SSTgCell.xib) 2.新建一个自定义的类 (自定义类需要继承自系统自带的view, 继承自哪个类, 取决于xib根对象的Class ...
- Android Studio开发Android应用如何签名
1.使用jdk自带的工具生成keystore 使用cmd命令行进入到jdk的bin目录(比如:C:\Program Files\Java\jdk1.7.0_01\bin) 运行如下命令: C:\Pro ...
- CJCMS系列---说说项目中的缓存实现(1)
缓存者,临时文件交换区也.主要就是方便查找,提高查找效率(效率在于读内存速度比读硬盘快). 大多数的项目的缓存都是通过设定过期时间来做的,可是我对于这样的替换策略不以为然,而且会导致混乱. 有人说: ...
- jquery easyui tree动态加载子节点
1.前端tree绑定时,使用onBeforeExpand事件:当节点展开时触发加载子节点,自动会向服务端发送请求:url为绑定url,参数为当前节点id this.tree = { method: ' ...
- virut详细分析
Virut分析 0x00.综合描述 virut样本的执行过程大体可以分为六步:第一步,解密数据代码,并调用解密后的代码:第二步,通过互斥体判断系统环境,解密病毒代码并执行:第三步,创建内存映射文件,执 ...
- C#高级编程笔记 2016年10月26日 MVC入门 Controller
1.MVC的定义: Models: Classes that represent the data of the application and that use validation logi ...
- phpstudy 80端口被占用,修改端口
搭建mantis,总会出现80端口被占用的情况.看到别的步骤是:1.cmd 运行netstat -ano查看80端口被什么占用,然后在任务管理器找到对应的结束进程.通常情况下是被System占用,右击 ...
- Python~字典
if not isinstance(x, (int, float)): raise TypeError('bad operand type') range() raw_input(‘birth’) ...
- Android安全攻防战,反编译与混淆技术完全解析(上)
转载请注明出处:http://blog.csdn.net/guolin_blog/article/details/49738023 之前一直有犹豫过要不要写这篇文章,毕竟去反编译人家的程序并不是什么值 ...