大连网站制作建设,怎么制作一个平台,wordpress无法开始安装,茌平网页定制Redis简单案例(四) Session的管理 原文:Redis简单案例(四) Session的管理负载均衡#xff0c;这应该是一个永恒的话题#xff0c;也是一个十分重要的话题。毕竟当网站成长到一定程度#xff0c;访问量自然也是会跟着增长#xff0c;这个时候#xff0c; 一般都会对其进行负… Redis简单案例(四) Session的管理 原文: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 下面是演示控制器的具体代码 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?sessionNamesessionName);
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 classrow5 div classcol-md-66 form methodpost action/7 div classform-group8 labelsession name/label9 input typetext namesessionName /
10 /div
11 div classform-group
12 labelsession value/label
13 input typetext namesessionValue /
14 /div
15 button typesubmitset session/button
16 /form
17 /div
18 /div
19 div classrow
20 div classcol-md-6
21 p
22 site: ViewBag.Site
23 /p
24 /div
25 /div Index.cshtml 1 {
2 ViewData[Title] About;
3 }
4 pViewBag.Session /p
5 psite: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.Common7 {8 public class RedisSession : ISession9 {
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 IEnumerablestring 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(60));
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 SessionExtension2 {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.AddSingletonISession, RedisSession(); 下面是修改之后控制器的代码 1 using AutoCompleteDemo.Common;2 using Microsoft.AspNetCore.Http;3 using Microsoft.AspNetCore.Mvc;4 5 namespace AutoCompleteDemo.Controllers6 {7 public class SessionController : Controller8 {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?sessionNamesessionName);
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 posted on 2018-05-21 16:50 NET未来之路 阅读(...) 评论(...) 编辑 收藏 转载于:https://www.cnblogs.com/lonelyxmas/p/9068034.html