简述

  基于Memcache的Session大家都各有各的说法,比方说:当memcached集群发生故障(比如内存溢出)或者维护(比如升级、增加或减少服务器)时,用户会无法登录,或者被踢掉线等等,每种技术各有优缺点,只是适应的场景不同罢了。

知识点补充

  服务器Memcache配置:https://www.cnblogs.com/chenyanbin/p/11415368.html

  Memcache集群配置:https://www.cnblogs.com/chenyanbin/p/11441490.html

  Mvc校验用户是否登陆:https://www.cnblogs.com/chenyanbin/p/11397576.html

  演示代码使用的其他完整类库:https://www.cnblogs.com/chenyanbin/p/11186495.html

代码演示(.Net的Mvc架构):

登陆页控制器

         IBllSession bllSession = BllSessionFactory.GetCurrentBllSession(); //业务层基类
/// <summary>
/// 处理登陆的表单
/// </summary>
/// <returns></returns>
public ActionResult ProcessLogin()
{
try
{
string user_name = Request["LoginId"]; //用户名
string user_pwd = Request["LoginPwd"]; //密码
UserInfo model = new UserInfo(); //实体类
model.UName = user_name; //实体类赋值
model.UPwd = user_pwd;
if (bllSession.UserInfo.Select(model).Count > ) //判断用户名密码是否正确
{
//旧方法
//Session["loginUser"] = user_name; //新方法
//Memcache+Cookie替代Session登陆
//立即分配一个标志GUID,把标志作为Memcache存储数据的key,把用户对象放到Memcache,把GUID写到客户端cookie里面去
string userLoginId = Guid.NewGuid().ToString(); //生成一个随机GUID
//将用户的数据写入Memcache
MemcacheHelper.AddCache(userLoginId, user_name, DateTime.Now.AddMinutes()); //Memcache帮助类
//往客户端写入Cookie
Response.Cookies["userLoginId"].Value= userLoginId; //将GUID写入Cookie
return Content("ok");
}
else
{
return Content("用户名或密码错误!你会登陆吗?");
}
}
catch (Exception ex)
{
throw ex;
}
}

过滤器基类(判断用户是否登陆)

 using Sam.OA.Common;
using System;
using System.Web.Mvc; namespace Sam.OA.WEBAPP.Controllers
{
/// <summary>
/// 控制器基类帮助类
/// 作者:陈彦斌
/// 更新时间:2019年9月1日17:43:10
/// </summary>
public class BaseController:Controller
{
public bool IsCheckedUserLogin = true;
protected override void OnActionExecuted(ActionExecutedContext filterContext)
{
base.OnActionExecuted(filterContext);
//校验用户是否已登录
if (IsCheckedUserLogin )
{
//新方法
//使用Memcache+Cookie代替Session
if (Request.Cookies["userLoginId"] == null)
{
filterContext.HttpContext.Response.Redirect("/UserLogin/Index");
return;
}
string userGuid = Request.Cookies["userLoginId"].Value; //拿到用户的GUID
object obj = MemcacheHelper.GetCache(userGuid);
if (obj == null || obj.ToString() == "")
{
//用户长时间不操作,超时
filterContext.HttpContext.Response.Redirect("/UserLogin/Index");
}
//滑动窗口机制
MemcacheHelper.SetCache(userGuid, obj, DateTime.Now.AddMinutes()); //旧方法Session
//if (filterContext.HttpContext.Session["loginUser"] == null)
//{
// filterContext.HttpContext.Response.Redirect("/UserLogin/Index");
//}
}
}
}
}

 Memcache帮助类(MemcacheHelper.cs)

 using Memcached.ClientLibrary;
using System; namespace Sam.OA.Common
{
/// <summary>
/// Memcache缓存帮助类
/// 作者:陈彦斌
/// 时间:2019年9月1日15:48:12
/// </summary>
public sealed class MemcacheHelper
{
private static MemcachedClient memcachedClient;
static MemcacheHelper()
{
//分布式Memcached服务器ip 端口
string strAppMemcached = DbUtil.memcacheServiceList;
if (strAppMemcached==""|| strAppMemcached==null)
{
throw new Exception("Memcache远程服务器Ip和端口未配置");
}
string[] servers = strAppMemcached.Split(','); //Memcache机器IP
//初始化池
SockIOPool pool = SockIOPool.GetInstance();
pool.SetServers(servers); //关联连接池
pool.InitConnections = ; //初始化链接
pool.MinConnections = ; //最小连接数
pool.MaxConnections = ; //最大连接数
pool.SocketConnectTimeout = ; //Socket超时连接时间
pool.SocketTimeout = ; //Socket超时时间
pool.MaintenanceSleep = ; //Socket休眠时间
pool.Failover = true;
pool.Nagle = false;
pool.Initialize(); //初始化
//客户端实例
if (memcachedClient == null)
{
memcachedClient = new MemcachedClient();
}
memcachedClient.EnableCompression = false; //启动压缩
}
/// <summary>
/// 获取Memcache缓存数据
/// </summary>
/// <param name="CacheKey">键</param>
/// <returns></returns>
public static object GetCache(string CacheKey)
{
return memcachedClient.Get(CacheKey);
}
/// <summary>
/// 设置Memcache缓存数据
/// </summary>
/// <param name="CacheKey">键</param>
/// <param name="CacheValue">值</param>
public static void AddCache(string CacheKey, object CacheValue)
{
memcachedClient.Add(CacheKey, CacheValue);
}
/// <summary>
/// 设置Memcache缓存数据
/// </summary>
/// <param name="CacheKey">键</param>
/// <param name="CacheValue">值</param>
/// <param name="expDate">过期时间</param>
public static void AddCache(string CacheKey, object CacheValue, DateTime expDate)
{
memcachedClient.Add(CacheKey, CacheValue,expDate);
}
/// <summary>
/// 设置Memcache缓存数据,key存在则更新,否则新增
/// </summary>
/// <param name="CacheKey">键</param>
/// <param name="CacheValue">值</param>
public static void SetCache(string CacheKey, object CacheValue)
{
memcachedClient.Set(CacheKey, CacheValue);
}
/// <summary>
/// 设置Memcache缓存数据,key存在则更新,否则新增
/// </summary>
/// <param name="CacheKey">键</param>
/// <param name="CacheValue">值</param>
/// <param name="expDate">过期时间</param>
public static void SetCache(string CacheKey, object CacheValue, DateTime expDate)
{
memcachedClient.Set(CacheKey, CacheValue, expDate);
}
}
}

觉得对你有帮助的话,帮忙推荐下,还有不懂的地方,欢迎下方留言,明天继续更新Redis!~~

.Net 基于Memcache集群的分布式Session的更多相关文章

  1. spring boot:使用redis cluster集群作为分布式session(redis 6.0.5/spring boot 2.3.1)

    一,为什么要使用分布式session? HpptSession默认使用内存来管理Session,如果将应用横向扩展将会出现Session共享问题, 所以我们在创建web集群时,把session保存到r ...

  2. 负载均衡集群中的session解决方案【转】

    通常面临的问题 从用户端来解释,就是当一个用户第一次访问被负载均衡代理到后端服务器A并登录后,服务器A上保留了用户的登录信息:当用户再次发送请求时, 根据负载均衡策略可能被代理到后端不同的服务器,例如 ...

  3. redis 与java的连接 和集群环境下Session管理

    redis 的安装与设置开机自启(https://www.cnblogs.com/zhulina-917/p/11746993.html)  第一步: a) 搭建环境 引入 jedis jar包 co ...

  4. nginx负载均衡集群中的session共享说明

    在网站使用nginx+php做负载均衡情况下,同一个IP访问同一个页面会被分配到不同的服务器上,如果session不同步的话,就会出现很多问题,比如说最常见的登录状态. 下面罗列几种nginx负载均衡 ...

  5. C# Memcache集群原理、客户端配置详细解析

    概述 memcache是一套开放源的分布式高速缓存系统.由服务端和客户端组成,以守护程序(监听)方式运行于一个或多个服务器中,随时会接收客户端的连接和操作.memcache主要把数据对象缓存到内存中, ...

  6. Nginx+PHP负载均衡集群环境中Session共享方案 - 运维笔记

    在网站使用nginx+php做负载均衡情况下,同一个IP访问同一个页面会被分配到不同的服务器上,如果session不同步的话,就会出现很多问题,比如说最常见的登录状态. 下面罗列几种nginx负载均衡 ...

  7. redis内存分配管理与集群环境下Session管理

    ##################内存管理############### 1.Redis的内存管理 .与memcache不同,没有实现自己的内存池 .在2..4以前,默认使用标准的内存分配函数(li ...

  8. 解决应用服务器变为集群后的Session问题

    2.2.4.2 解决应用服务器变为集群后的Session问题 先来看一下什么是Session. 用户使用网站的服务,基本上需要浏览器与Web 服务器的多次交互.HTTP 协议本身是无状态的,需要基于H ...

  9. Quartz集成springMVC 的方案二(持久化任务、集群和分布式)

    Quartz是一个开放源码项目,专注于任务调度器,提供了极为广泛的特性如持久化任务,集群和分布式任务等. Quartz核心是调度器,还采用多线程管理. 1.持久化任务:当应用程序停止运行时,所有调度信 ...

随机推荐

  1. codemirror使用

    JS使用 使用bower下载 bower i codemirror 引入样式文件 <link rel="stylesheet" type="text/css&quo ...

  2. docker学习ppt

    保存下学习资料

  3. CentOS7源码安装Nginx

    系统平台:腾讯云服务器 CentOS 7.3 64位 一.安装编译工具及库文件 [root@VM_0_5_centos ~]# yum install -y make zlib zlib-devel ...

  4. JavaScript基础学习第六天

    目标: 能够使用对象的方式处理数据 ☞ 代码预解析: 1. 变量提升 :当程序中遇到定义变量后,就会将该变量的定义提升到当前作用域的开始位置,不包括变量的赋值 2. 函数提升:当程序中遇到函数的声明时 ...

  5. 解决axios接收二进制流文件乱码问题

    1. 须将axios 配置中的responseType设置为'arraybuffer',这样就不会让表格出现乱码现象: 2. 如果要动态设置文件名则需要让后台将名字设置到响应头中,否则将是一个乱码的文 ...

  6. DevOps相关知识点

    DevOps 持续集成 简述 持续集成简称CI,是软件的开发和发布标准流程的最重要的部分 作为一个开发实践,在C中可以通过自动化等手段高频地去获取产品反馈并响应反馈的过程 简单的来说,持续集成就是持续 ...

  7. poj 1455 Crazy tea party

    这道题第一眼看去很难,其实不然,短短几行代码就搞定了. 说一下大概思路,如果是排成一排的n个人,如 1 2 3 4 5 6 7 8 我们要变成 8 7 6 5 4 3 2 1 需要交换 28次,找规律 ...

  8. go 学习笔记之工作空间

    搭建好 Go 的基本环境后,现在可以正式开始 Go 语言的学习之旅,初学时建议在默认的 GOPATH 工作空间规范编写代码,基本目录结构大概是这个样子. . |-- bin | `-- hello.e ...

  9. FormLayout and FormData

    FormLayout通过为小窗口部件创建四边的Form附加值(attachment)来进行工作,并且把这些Form附加值存储在布局数据中.一个附加值让一个小窗口部件指定的一边粘贴(attach)到父C ...

  10. Wpf窗口设置可拖动

    在窗口界面的一个控件(TopGrid)设置如下MouseLeftButtonDown事件即可 private void TopGrid_MouseLeftButtonDown(object sende ...