Redis简单案例(四) Session的管理
负载均衡,这应该是一个永恒的话题,也是一个十分重要的话题。毕竟当网站成长到一定程度,访问量自然也是会跟着增长,这个时候,
一般都会对其进行负载均衡等相应的调整。现如今最常见的应该就是使用Nginx来进行处理了吧。当然Jexus也可以达到一样的效果。既然是
负载均衡,那就势必有多台服务器,如果不对session进行处理,那么就会造成Session丢失的情况。有个高大上的名字叫做分布式Session。
举个通俗易懂的例子,假设现在有3台服务器做了负载,用户在登陆的时候是在a服务器上进行的,此时的session是写在a服务器上的,那
么b和c两台服务器是不存在这个session的,当这个用户进行了一个操作是在b或c进行处理的,而且这个操作是要登录后才可以的,那么就会
背景交待完毕,简单的实践一下。
相关技术 | 说明 |
ASP.NET Core | 演示的两个站点所用的技术 |
Redis | 用做Session服务器 |
Nginx/Jexus | 用做反向代理服务器,演示主要用了Nginx,最后也介绍了Jexus的用法 |
IIS/Jexus | 用做应用服务器,演示用了本地的IIS,想用Jexus来部署可参考前面的相关文章 |
先来看看不进行Session处理的做法,看看Session丢失的情况,然后再在其基础上进行改善。
在ASP.NET Core中,要使用session需要在Startup中的ConfigureServices添加 services.AddSession(); 以及在Configure中添加
app.UseSession(); 才能使用。在控制器中的用法就是 HttpContext.Session.XXX ,下面是演示控制器的具体代码:
[HttpGet("/")]
[ResponseCache(NoStore =true)]
public IActionResult Index()
{
ViewBag.Site = "site 1";
return View();
}
[HttpPost("/")]
public IActionResult Index(string sessionName,string sessionValue)
{
//set the session
HttpContext.Session.Set(sessionName,System.Text.Encoding.UTF8.GetBytes(sessionValue));
return Redirect("/about?sessionName="+sessionName);
} [HttpGet("/about")]
[ResponseCache(NoStore = true)]
public IActionResult About(string sessionName)
{
byte[] bytes;
ViewBag.Site = "site 1";
//get the session
if (HttpContext.Session.TryGetValue(sessionName, out bytes))
{
ViewBag.Session = System.Text.Encoding.UTF8.GetString(bytes);
}
else
{
ViewBag.Session = "empty";
}
return View();
}
其中的ViewBag.Site是用来标识当前访问的是那个负载的站点。这用就不用去查日记访问了那个站点了,直接在页面上就能看到了。从
Session的用法也看出了与之前的有所不同,Session的值是用byte存储的。我们可以写个扩展方法把它封装一下,这样就方便我们直接向之
视图比较简单,一个写Session,一个读Session。Index.cshtml用于填写Session的信息,提交后跳转到About.cshtml。
@{
ViewData["Title"] = "Home Page";
}
<div class="row">
<div class="col-md-6">
<form method="post" action="/">
<div class="form-group">
<label>session name</label>
<input type="text" name="sessionName" />
</div>
<div class="form-group">
<label>session value</label>
<input type="text" name="sessionValue" />
</div>
<button type="submit">set session</button>
</form>
</div>
</div>
<div class="row">
<div class="col-md-6">
<p>
site: @ViewBag.Site
</p>
</div>
</div>
Index.cshtml
@{
ViewData["Title"] = "About";
}
<p>@ViewBag.Session </p>
<p>site:@ViewBag.Site</p>
About.cshtml
到这里,我们是已经把我们要的“网站”给开发好了,下面是把这个“网站”部署到IIS上面。我们要在IIS上部署两个站点,这两个站点用于
这时可以参考dotNET Core的文档,至于为什么没有放到Linux下呢,毕竟是台老电脑了,开多个虚拟机电脑吃不消,云服务器又还没想好要
租那家的,所以只好放到本地的IIS上来演示了,想在Linux下部署ASP.NET Core可以参考我前面的博文,也是很简单的喔。
这是部署到本地IIS上面的两个站点,site1和site2。
Nginx的负载均衡模块,更多的细节可以去它的官网看一下。这里就不做详细的介绍,毕竟这些配置都十分的简单。
Nginx的配置也配好了,接下来就是启动我们的Nginx服务器,执行 /usr/local/nginx/sbin/nginx 即可,最后就是访问我们Nginx这个
空壳站点http://192.168.198.128:8033(实际是访问我们在IIS上的那2个站点),然后就可以看看效果了,建议把浏览器的缓存禁用掉,不然
轮询的效果可能会出不来。
可以看到轮询的效果已经出来了,访问Linux下面的Nginx服务器,实际上是访问IIS上的site1和site2。我们是在站点2 设置了session,
但是在站点2却得不到这个session值,而是在站点1才能得到这个值。这是因为我们用的算法是Nginx默认的轮询算法,也就是说它是一直这样
循环访问我们的站点1和站点2,站点1->站点2 ->站点1->站点2....,演示是在站点2设置Session并提交,但它是提交到了站点1去执行,执行
完成后Redirect到了站点2,所以会看到站点2上没有session的信息而站点1上面有。
好了,警报提醒,Session丢失了,接下来我们就要想办法处理了这个常见并且棘手的问题了, 本文的处理方法是用Redis做一台单独的
Session服务器,用这台服务器来统一管理我们的Session,当然这台Redis服务器会做相应的持久化配置以及主从或Cluster集群,毕竟没人能
保证这台服务器不出故障。思路图如下:
在上面例子的基础上,添加一个RedisSession类,用于处理Session,让其继承ISession接口
using Microsoft.AspNetCore.Http;
using System;
using System.Collections.Generic;
using System.Threading.Tasks; namespace AutoCompleteDemo.Common
{
public class RedisSession : ISession
{
private IRedis _redis;
public RedisSession(IRedis redis)
{
_redis = redis;
} public string Id
{
get
{
return Guid.NewGuid().ToString();
}
} public bool IsAvailable
{
get
{
throw new NotImplementedException();
}
} public IEnumerable<string> Keys
{
get
{
throw new NotImplementedException();
}
} public void Clear()
{
throw new NotImplementedException();
} public Task CommitAsync()
{
throw new NotImplementedException();
} public Task LoadAsync()
{
throw new NotImplementedException();
} public void Remove(string key)
{
_redis.Del(key);
} public void Set(string key, byte[] value)
{
_redis.Set(key, System.Text.Encoding.UTF8.GetString(value),TimeSpan.FromSeconds());
} public bool TryGetValue(string key, out byte[] value)
{ string res = _redis.Get(key);
if (string.IsNullOrWhiteSpace(res))
{
value = null;
return false;
}
else
{
value = System.Text.Encoding.UTF8.GetBytes(res);
return true;
}
}
}
}
会有一个过期的时间,这里默认给了60秒,真正实践的时候可能要结合SessionOptions来进行修改这里的代码。前面也提到写个扩展方法,可以减少
调用的代码量和方便我们的使用,所以还写了一个对Session的扩展,方便在控制器中使用,这样就不用每次都把要存的东西再处理成byte。
public static class SessionExtension
{
public static string GetExtension(this ISession session, string key)
{
string res = string.Empty;
byte[] bytes;
if (session.TryGetValue(key, out bytes))
{
res = System.Text.Encoding.UTF8.GetString(bytes);
}
return res;
}
public static void SetExtension(this ISession session, string key,string value)
{
session.Set(key, System.Text.Encoding.UTF8.GetBytes(value));
}
}
下面是修改之后控制器的代码:
using AutoCompleteDemo.Common;
using Microsoft.AspNetCore.Http;
using Microsoft.AspNetCore.Mvc; namespace AutoCompleteDemo.Controllers
{
public class SessionController : Controller
{
private ISession _session;
public SessionController(ISession session)
{
_session = session;
} [HttpGet("/")]
[ResponseCache(NoStore =true)]
public IActionResult Index()
{
ViewBag.Site = "site 1";
return View();
}
[HttpPost("/")]
public IActionResult Index(string sessionName,string sessionValue)
{
//set the session
_session.SetExtension(sessionName, sessionValue);
return Redirect("/about?sessionName="+sessionName);
} [HttpGet("/about")]
[ResponseCache(NoStore = true)]
public IActionResult About(string sessionName)
{
//get the session
ViewBag.Session = _session.GetExtension(sessionName);
ViewBag.Site = "site 1";
return View();
}
}
}
也整洁了不少。是直接用了自己写的扩展方法。
视图没有变化。Nginx的配置也没有变化。下面是对session进行一番简单处理后的效果。
可以看到无论在那个站点,都能正常的读取到session服务器里面的值。也就是说,经过简单的初步处理,我们的Session在负载均衡下面已经
不会丢失了。当然这个只能说是一个雏形,还有更多的细节要去完善。
文中讲到用Jexus也可以完成同样的功能,下面就简单说一下它的配置:

这样就可以完成和上面演示中同样的功能。
当然,对于分布式Session的管理,这只是其中一种解决方法--基于Redis的解决方案,还有许多前人总结出来的方案,好比孤独侠客前辈的
这篇博客总结了6种方案:http://www.cnblogs.com/lonely7345/p/3796488.html,都是值得我们这些小辈去学习和研究的。
源码已上传到github:
https://github.com/hwqdt/Demos/tree/master/src/RedisDemo
Redis简单案例(四) Session的管理的更多相关文章
- SpringDataRedis操作Redis简单案例
Jedis Jedis是Redis官方推出的一款面向Java的客户端,提供了很多接口供Java语言调用.可以在Redis官网下载,当然还有一些开源爱好者提供的客户端,如Jredis.SRP等等,推荐使 ...
- Redis简单案例(三) 连续登陆活动的简单实现
连续登陆活动,或许大家都不会陌生,简单理解就是用户连续登陆了多少天之后,系统就会送一些礼品给相应的用户.最常见的 莫过于游戏和商城这些.游戏就送游戏币之类的东西,商城就送一些礼券.正值国庆,应该也有不 ...
- Redis简单案例(二) 网站最近的访问用户
我们有时会在网站中看到最后的访问用户.最近的活跃用户等等诸如此类的一些信息.本文就以最后的访问用户为例, 用Redis来实现这个小功能.在这之前,我们可以先简单了解一下在oracle.sqlserve ...
- Redis简单案例(一) 网站搜索的热搜词
对于一个网站来说,无论是商城网站还是门户网站,搜索框都是有一个比较重要的地位,它的存在可以说是 为了让用户更快.更方便的去找到自己想要的东西.对于经常逛这个网站的用户,当然也会想知道在这里比较“火” ...
- [redis] session 保存到 redis 简单实现
参考资料: [session保存到redis简单实现]http://blog.csdn.net/ppt0501/article/details/46700221 [Redis学习]http://blo ...
- redis学习教程四《管理、备份、客户端连接》
redis学习教程四<管理.备份.客户端连接> 一:Redis服务器命令 Redis服务器命令 下表列出了与Redis服务器相关的一些基本命令. 序号 命令 说明 1 BGREWRITE ...
- Nginx+Tomcat关于Session的管理
前言 Nginx+Tomcat对Session的管理一直有了解,但是一直没有实际操作一遍,本文从最简单的安装启动开始,通过实例的方式循序渐进的介绍了几种管理session的方式. nginx安装配置 ...
- Redis安装及实现session共享
一.Redis介绍 1.redis是key-value的存储系统,属于非关系型数据库 2.特点:支持数据持久化,可以让数据在内存中保存到磁盘里(memcached:数据存在内存里,如果服务重启,数据会 ...
- SSM框架整合(Spring+SrpingMVC+Mybatis) 简单案例
简介: SSM框架是Spring,SpringMVC 和Mybatis框架的整合,是标准的MVC模式,将整个系统划分为表现层,controller层,service层,dao层四层. Spring实现 ...
随机推荐
- Zabbix基本配置及监控主机
监控主机一版需要在被监控的主机上安装Zabbix Agent 监控主机 安装zabbix-agent 首先需要在被监控的主机上安装agent,可以下载预编译好的RPM进行安装,下载地址:http:// ...
- Javascript实用方法
这篇我主要记录一些在工作中常用的.实用的方法. String trim 字符串方法中的trim主要用来去空格使用,很多时候,在后台做参数处理的时候,我们都会使用该方法,比如在获取用户输入的账户时 va ...
- CRL快速开发框架系列教程十(导出对象结构)
本系列目录 CRL快速开发框架系列教程一(Code First数据表不需再关心) CRL快速开发框架系列教程二(基于Lambda表达式查询) CRL快速开发框架系列教程三(更新数据) CRL快速开发框 ...
- AFNetworking 3.0 源码解读(十一)之 UIButton/UIProgressView/UIWebView + AFNetworking
AFNetworking的源码解读马上就结束了,这一篇应该算是倒数第二篇,下一篇会是对AFNetworking中的技术点进行总结. 前言 上一篇我们总结了 UIActivityIndicatorVie ...
- CSS知识总结(八)
CSS常用样式 8.变形样式 改变元素的大小,透明,旋转角度,扭曲度等. transform : none | <transform-function> <transform-fun ...
- VS项目中使用Nuget还原包后编译生产还一直报错?
Nuget官网下载Nuget项目包的命令地址:https://www.nuget.org/packages 今天就遇到一个比较奇葩的问题,折腾了很久终于搞定了: 问题是这样的:我的解决方案原本是好好的 ...
- [转载]SQL Server 2008 R2安装时选择的是windows身份验证,未选择混合身份验证的解决办法
安装过程中,SQL Server 数据库引擎设置为 Windows 身份验证模式或 SQL Server 和 Windows 身份验证模式.本文介绍如何在安装后更改安全模式. 如果在安装过程中选择&q ...
- [转载]敏捷开发之Scrum扫盲篇
现在敏捷开发是越来越火了,人人都在谈敏捷,人人都在学习Scrum和XP... 为了不落后他人,于是我也开始学习Scrum,今天主要是对我最近阅读的相关资料,根据自己的理解,用自己的话来讲述S ...
- Ubuntu下开启php调试模式,显示报错信息
在Ubuntu下php的缺省设置是不显示错误信息的,如果程序出错会显示“无法处理此请求的错误提示”,这在开发环境下非常不方便. 其实我们只要编辑下apache的配置文件就好 1.我的apache 配置 ...
- FineReport:任意时刻只允许在一个客户端登陆账号的插件
在使用FineReport报表系统中,处于账户安全考虑,有些企业希望同一账号在任意时刻智能在统一客户端登录.那么当A用户在C1客户端登陆后,该账号又在另外一个C2客户端登陆,服务器如何取判断呢? 开发 ...