负载均衡,这应该是一个永恒的话题,也是一个十分重要的话题。毕竟当网站成长到一定程度,访问量自然也是会跟着增长,这个时候,

一般都会对其进行负载均衡等相应的调整。现如今最常见的应该就是使用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 ,下面是演示控制器的具体代码:

  1.     [HttpGet("/")]
  2. [ResponseCache(NoStore =true)]
  3. public IActionResult Index()
  4. {
  5. ViewBag.Site = "site 1";
  6. return View();
  7. }
  8. [HttpPost("/")]
  9. public IActionResult Index(string sessionName,string sessionValue)
  10. {
  11. //set the session
  12.     HttpContext.Session.Set(sessionName,System.Text.Encoding.UTF8.GetBytes(sessionValue));
  13. return Redirect("/about?sessionName="+sessionName);
  14. }
  15.  
  16. [HttpGet("/about")]
  17. [ResponseCache(NoStore = true)]
  18. public IActionResult About(string sessionName)
  19. {
  20. byte[] bytes;
  21. ViewBag.Site = "site 1";
  22. //get the session
  23. if (HttpContext.Session.TryGetValue(sessionName, out bytes))
  24. {
  25. ViewBag.Session = System.Text.Encoding.UTF8.GetString(bytes);
  26. }
  27. else
  28. {
  29. ViewBag.Session = "empty";
  30. }
  31. return View();
  32. }

  其中的ViewBag.Site是用来标识当前访问的是那个负载的站点。这用就不用去查日记访问了那个站点了,直接在页面上就能看到了。从

Session的用法也看出了与之前的有所不同,Session的值是用byte存储的。我们可以写个扩展方法把它封装一下,这样就方便我们直接向之

前一样的写法,不用每次都转成byte再进行读写了。

  视图比较简单,一个写Session,一个读Session。Index.cshtml用于填写Session的信息,提交后跳转到About.cshtml。

  1. @{
  2. ViewData["Title"] = "Home Page";
  3. }
  4. <div class="row">
  5. <div class="col-md-6">
  6. <form method="post" action="/">
  7. <div class="form-group">
  8. <label>session name</label>
  9. <input type="text" name="sessionName" />
  10. </div>
  11. <div class="form-group">
  12. <label>session value</label>
  13. <input type="text" name="sessionValue" />
  14. </div>
  15. <button type="submit">set session</button>
  16. </form>
  17. </div>
  18. </div>
  19. <div class="row">
  20. <div class="col-md-6">
  21. <p>
  22. site: @ViewBag.Site
  23. </p>
  24. </div>
  25. </div>

Index.cshtml

  1. @{
  2. ViewData["Title"] = "About";
  3. }
  4. <p>@ViewBag.Session </p>
  5. <p>site:@ViewBag.Site</p>

About.cshtml

  到这里,我们是已经把我们要的“网站”给开发好了,下面是把这个“网站”部署到IIS上面。我们要在IIS上部署两个站点,这两个站点用于

我们负载均衡的使用。两个站点的区分就是ViewBag.Site,一个显示site1,一个显示site2。ASP.NET Core在IIS上部署可能不会太顺畅,

这时可以参考dotNET Core的文档,至于为什么没有放到Linux下呢,毕竟是台老电脑了,开多个虚拟机电脑吃不消,云服务器又还没想好要

租那家的,所以只好放到本地的IIS上来演示了,想在Linux下部署ASP.NET Core可以参考我前面的博文,也是很简单的喔。

  这是部署到本地IIS上面的两个站点,site1和site2。

  

  站点我们是已经部署OK了,还是要先检查一下这两个站点是否能正常访问。如果这两个不能正常访问,那么我们下面的都是。。。

  

  OK!能正常访问,接下来就是今天下一个主角Nginx登场的时候了。用法很简单,下面给出主要的配置,主要的模块是upstream,这个是

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接口  

  1. using Microsoft.AspNetCore.Http;
  2. using System;
  3. using System.Collections.Generic;
  4. using System.Threading.Tasks;
  5.  
  6. namespace AutoCompleteDemo.Common
  7. {
  8. public class RedisSession : ISession
  9. {
  10. private IRedis _redis;
  11. public RedisSession(IRedis redis)
  12. {
  13. _redis = redis;
  14. }
  15.  
  16. public string Id
  17. {
  18. get
  19. {
  20. return Guid.NewGuid().ToString();
  21. }
  22. }
  23.  
  24. public bool IsAvailable
  25. {
  26. get
  27. {
  28. throw new NotImplementedException();
  29. }
  30. }
  31.  
  32. public IEnumerable<string> Keys
  33. {
  34. get
  35. {
  36. throw new NotImplementedException();
  37. }
  38. }
  39.  
  40. public void Clear()
  41. {
  42. throw new NotImplementedException();
  43. }
  44.  
  45. public Task CommitAsync()
  46. {
  47. throw new NotImplementedException();
  48. }
  49.  
  50. public Task LoadAsync()
  51. {
  52. throw new NotImplementedException();
  53. }
  54.  
  55. public void Remove(string key)
  56. {
  57. _redis.Del(key);
  58. }
  59.  
  60. public void Set(string key, byte[] value)
  61. {
  62. _redis.Set(key, System.Text.Encoding.UTF8.GetString(value),TimeSpan.FromSeconds());
  63. }
  64.  
  65. public bool TryGetValue(string key, out byte[] value)
  66. {
  67.  
  68. string res = _redis.Get(key);
  69. if (string.IsNullOrWhiteSpace(res))
  70. {
  71. value = null;
  72. return false;
  73. }
  74. else
  75. {
  76. value = System.Text.Encoding.UTF8.GetBytes(res);
  77. return true;
  78. }
  79. }
  80. }
  81. } 
  ISession接口定义了不少东西,这里只实现了ISession中的部分内容,主要的Set和Get实现了,因为演示用不到那么多~~,就偷偷懒。Session

会有一个过期的时间,这里默认给了60秒,真正实践的时候可能要结合SessionOptions来进行修改这里的代码。前面也提到写个扩展方法,可以减少

调用的代码量和方便我们的使用,所以还写了一个对Session的扩展,方便在控制器中使用,这样就不用每次都把要存的东西再处理成byte。

  1. public static class SessionExtension
  2. {
  3. public static string GetExtension(this ISession session, string key)
  4. {
  5. string res = string.Empty;
  6. byte[] bytes;
  7. if (session.TryGetValue(key, out bytes))
  8. {
  9. res = System.Text.Encoding.UTF8.GetString(bytes);
  10. }
  11. return res;
  12. }
  13. public static void SetExtension(this ISession session, string key,string value)
  14. {
  15. session.Set(key, System.Text.Encoding.UTF8.GetBytes(value));
  16. }
  17. }   
  要使用刚才定义的RedisSession,还需要在Startup的ConfigureServices中添加下面这行代码。
 services.AddSingleton<ISession, RedisSession>();  

  下面是修改之后控制器的代码:

  1. using AutoCompleteDemo.Common;
  2. using Microsoft.AspNetCore.Http;
  3. using Microsoft.AspNetCore.Mvc;
  4.  
  5. namespace AutoCompleteDemo.Controllers
  6. {
  7. public class SessionController : Controller
  8. {
  9. private ISession _session;
  10. public SessionController(ISession session)
  11. {
  12. _session = session;
  13. }
  14.  
  15. [HttpGet("/")]
  16. [ResponseCache(NoStore =true)]
  17. public IActionResult Index()
  18. {
  19. ViewBag.Site = "site 1";
  20. return View();
  21. }
  22. [HttpPost("/")]
  23. public IActionResult Index(string sessionName,string sessionValue)
  24. {
  25. //set the session
  26. _session.SetExtension(sessionName, sessionValue);
  27. return Redirect("/about?sessionName="+sessionName);
  28. }
  29.  
  30. [HttpGet("/about")]
  31. [ResponseCache(NoStore = true)]
  32. public IActionResult About(string sessionName)
  33. {
  34. //get the session
  35. ViewBag.Session = _session.GetExtension(sessionName);
  36. ViewBag.Site = "site 1";
  37. return View();
  38. }
  39. }
  40. }  
  通过构造函数注入我们的ISession。然后就能使用我们自己定义的方法了,这种做法在ASP.NET Core中是随处可见的。而且控制器中的代码

也整洁了不少。是直接用了自己写的扩展方法。

  视图没有变化。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的管理的更多相关文章

  1. SpringDataRedis操作Redis简单案例

    Jedis Jedis是Redis官方推出的一款面向Java的客户端,提供了很多接口供Java语言调用.可以在Redis官网下载,当然还有一些开源爱好者提供的客户端,如Jredis.SRP等等,推荐使 ...

  2. Redis简单案例(三) 连续登陆活动的简单实现

    连续登陆活动,或许大家都不会陌生,简单理解就是用户连续登陆了多少天之后,系统就会送一些礼品给相应的用户.最常见的 莫过于游戏和商城这些.游戏就送游戏币之类的东西,商城就送一些礼券.正值国庆,应该也有不 ...

  3. Redis简单案例(二) 网站最近的访问用户

    我们有时会在网站中看到最后的访问用户.最近的活跃用户等等诸如此类的一些信息.本文就以最后的访问用户为例, 用Redis来实现这个小功能.在这之前,我们可以先简单了解一下在oracle.sqlserve ...

  4. Redis简单案例(一) 网站搜索的热搜词

    对于一个网站来说,无论是商城网站还是门户网站,搜索框都是有一个比较重要的地位,它的存在可以说是 为了让用户更快.更方便的去找到自己想要的东西.对于经常逛这个网站的用户,当然也会想知道在这里比较“火” ...

  5. [redis] session 保存到 redis 简单实现

    参考资料: [session保存到redis简单实现]http://blog.csdn.net/ppt0501/article/details/46700221 [Redis学习]http://blo ...

  6. redis学习教程四《管理、备份、客户端连接》

    redis学习教程四<管理.备份.客户端连接>  一:Redis服务器命令 Redis服务器命令 下表列出了与Redis服务器相关的一些基本命令. 序号 命令 说明 1 BGREWRITE ...

  7. Nginx+Tomcat关于Session的管理

    前言 Nginx+Tomcat对Session的管理一直有了解,但是一直没有实际操作一遍,本文从最简单的安装启动开始,通过实例的方式循序渐进的介绍了几种管理session的方式. nginx安装配置 ...

  8. Redis安装及实现session共享

    一.Redis介绍 1.redis是key-value的存储系统,属于非关系型数据库 2.特点:支持数据持久化,可以让数据在内存中保存到磁盘里(memcached:数据存在内存里,如果服务重启,数据会 ...

  9. SSM框架整合(Spring+SrpingMVC+Mybatis) 简单案例

    简介: SSM框架是Spring,SpringMVC 和Mybatis框架的整合,是标准的MVC模式,将整个系统划分为表现层,controller层,service层,dao层四层. Spring实现 ...

随机推荐

  1. pt-mext

    pt-mext实现的功能比较简单,就是将mysqladmin输出的多次迭代的相同status变量值放到同一行输出. 参数很少,除了--help和--version外,只有一个--relative参数 ...

  2. 从Vue.js窥探前端行业

    近年来前端开发趋势 1.旧浏览器逐渐淘汰,移动端需求增加: 旧浏览器主要指的是IE6-IE8,它是不支持ES5特性的:IE9+.chrome.sarafi.firefox对ES5是完全支持的,移动端大 ...

  3. 初学seaJs模块化开发,利用grunt打包,减少http请求

    原文地址:初学seaJs模块化开发,利用grunt打包,减少http请求 未压缩合并的演示地址:demo2 学习seaJs的模块化开发,适合对seajs基础有所了解的同学看,目录结构 js — —di ...

  4. Git分布式版本控制教程

    Git分布式版本控制Git 安装配置Linux&Unix平台 Debian/Ubuntu $ apt-get install git Fedora $ ) $ dnf and later) G ...

  5. 监控 SQL Server (2005/2008) 的运行状况

    Microsoft SQL Server 2005 提供了一些工具来监控数据库.方法之一是动态管理视图.动态管理视图 (DMV) 和动态管理函数 (DMF) 返回的服务器状态信息可用于监控服务器实例的 ...

  6. Raspberry Pi(树莓派)上安装Raspbian(无路由器,无显示器)

    一. 准备工作 1. 树莓派主板 型号:树莓派3 B型 处理器:四核64位ARM Cortex-A53 CPU 内核架构:ARMv8 2. 一张大于8G的TF卡(本人用的是32G的,也作为PiLFS用 ...

  7. 排序算法----调用库函数qsort进行快速排序

    功 能: 快速排序 头文件:stdlib.h 用 法: void qsort(void *base,int nelem,int width,int (*fcmp)(const void *,const ...

  8. MONO x64 amd_x64

    SharpPcap 是一个.NET 环境下的网络包捕获框架,基于著名的 pcap/WinPcap 库开发.提供了捕获.注入.分析和构建的功能,SharpPcap 还是一个跨平台的库,可以在Mono/. ...

  9. Concurrency != Parallelism

    前段时间在公司给大家分享GO语言的一些特性,然后讲到了并发概念,大家表示很迷茫,然后分享过程中我拿来了Rob Pike大神的Slides <Concurrency is not Parallel ...

  10. LINQ Group By操作

    在上篇文章 .NET应用程序与数据库交互的若干问题 这篇文章中,讨论了一个计算热门商圈的问题,现在在这里扩展一下,假设我们需要从两张表中统计出热门商圈,这两张表内容如下: 上表是所有政区,商圈中的餐饮 ...