先从web session的共享说起

 
许多系统需要提供7*24小时服务,这类系统肯定需要考虑灾备问题,单台服务器如果宕机可能无法立马恢复使用,这必定影响到服务。这个问题对于系统规模来说,从小到大可能面临的难度会相差很大。但对于原理来说其实就是需要准备备份系统随时可以替代正在服务的系统,也就是无论何时都有服务器可以提供服务。也就是灾备系统或者负载均衡。
 
提供灾备系统或者负载均衡系统都需要面临一个问题,那就是如何解决共享数据的问题。对于web服务器而言首先要解决的就是web session共享问题,比如A服务器的session如何可以在B服务器上也能一样使用呢?毕竟是物理隔离的两台服务器。
 
这方面的方案主要是两类:cookies和session共享。
 

cookies

这种方案的思路就是将session的数据写入到cookies里,每次请求的时候就可以带上信息,这样不管是哪台服务器都能得到同样的数据啦。这样不管换多少服务器都好处理。只不过这种方案需要在服务端开发时需要注意session的数据管理,而且需要接管session的生命周期。如果有一些老的系统可能session用的比较多,就不大好使了。而且将一些敏感数据写入cookies还要考虑安全问题,这对于一些数据敏感的系统也可能是个问题。
 
但如果能控制好session的数据这种方案个人觉得还是挺不错的,毕竟session并不适合存过多的数据。所以在我们的系统中是支持这种方案的,只需要打开开关参数就行。
 

session池化

还有一种方法就是把session共享出来,所有的服务器都连接到这个共享。这种方案可能是许多系统会使用的方案吧。因为将session池化,对于系统而言就变成透明了。程序员终于开心的将数据写入session咯。这种方案除了http服务器外,许多的tcp服务器也是类似的方案。
 
我们系统因为使用的java开发,使用tomcat时可以将session共享到memcached/redis中。而且这种操作完全不需要改动系统,直接在tomcat中配置即可。所以这种方案天然就支持啦。
 
 

做一个可扩展的缓存策略设计

原先的数据缓存都是放在jvm里的,所以机器多了每台服务器都要自己去加载缓存,这样一来命中就低。最近打算在系统里引入第三方缓存,当时在memcached和火的要死的redis里选择。现在来看每种内存产品都各有优势,如果硬生生的将现在这些老的缓存直接改成redis的如果以后需要用别的内存数据库又得大改代码。想到这就决定把缓存做一次设计,将现有的jvm缓存保留下来,然后做成策略以扩展新的缓存存储。
 
以前的许多缓存用的HashMap/ConcurrentHashMap,反正是键-对值。如果我们直接使用Map结构来作为缓存接口就可以不改变现有的一些代码,只需要改动缓存类内部的数据结构即可。这样的改动量就比较少。
 
比如原来的一些缓存单元结构:
public class RoleMenuCache implements IClearCache {
private static Map<String, RoleMenu> roleMenuCache = new HashMap<String, RoleMenu>();
...业务代码省略
}

这里主要是替换这个HashMap所以改动就比较小。

 
 
先来看看类图
 
 

Cachemanager

这个就是缓存的管理类,用于创建、释放缓存对象。这个类是各个所有缓存申请的入口。下面贴出来主要的代码:
public class CacheManager {
private final static Logger logger = LoggerFactory.getLogger(CacheManager.class);
private static Map<String, ICache> caches = new ConcurrentHashMap<>();
private static ICacheStrategy cacheStrategy = new DefaultCacheStategy();
private static String cacheStrategyClass; @SuppressWarnings("unchecked")
public static synchronized <T extends ICache> T getOrCreateCache(String cacheName, Class<?> keyClass, Class<?> valueCalss) {
T cache = (T) caches.get(cacheName);
if (cache != null) {
return cache;
}
cache = (T) cacheStrategy.createCache(cacheName, keyClass, valueCalss);
caches.put(cacheName, cache);
return cache;
} @SuppressWarnings("rawtypes")
public static synchronized void destroyCache(String cacheName) {
ICache cache = caches.remove(cacheName);
if (cache != null) {
cache.clear();
}
} }

ICache<K,V>

这个接口是规范缓存类的接口,所有的缓存类都要实现这个接口,而且它是继承java.util.Map接口的,这样就支持了Map派生的类,兼容老程序就好多了。
 

ICacheStrategy

对于具体的缓存实现就有一套策略,有一个ICacheStrategy接口来规范。这么一来,不管是jvm还是redis都可以自己单独扩展来实现。
public interface ICacheStrategy {
ICache createCache(String name, Class<?> keyClass, Class<?> valueCalss);
void destroyCache(ICache cache);
}
看一下DefaultCache的实现(代码只放了一部分主要的):
public class DefaultCache<K, V> implements ICache<K, V> {
protected Map<K, V> map;
private String name;
private long maxCacheSize;
private long maxLifetime;
private int cacheSize = 0; public DefaultCache(String name, long maxSize, long maxLifetime) {
this.name = name;
this.maxCacheSize = maxSize;
this.maxLifetime = maxLifetime;
map = new ConcurrentHashMap<K, V>(103);
} @Override
public V get(Object key) {
return map.get(key);
} @Override
public V put(K key, V value) {
return map.put(key, value);
} @Override
public V remove(Object key) {
return map.remove(key);
} @Override
public void putAll(Map<? extends K, ? extends V> m) {
map.putAll(m);
} @Override
public void clear() {
if (map != null) {
map.clear();
}
} }

对于调用方来说其实就很简单,只需要调用CacheManager即可,还是前面举的RoleMenuCache

例子,我们改造一下:
public class RoleMenuCache implements IClearCache {
private static Map<String, RoleMenu> roleMenuCache;
static {
roleMenuCache = CacheManager.getOrCreateCache("permissionCache", String.class, RoleMenu.class);
}
...业务代码省略
}

对于老代码的改造还是比较小的,而且这样的好处是以后想换成redis的也很简单,对于业务代码就不需要再修改了。

 
遇到Redis与泛型的问题
 
在扩展redis缓存策略的时候遇到一个问题,就是使用的jedis时,对于key值都是使用的string类型,这就给我们使用泛型设计留下了难题。当然为了兼容现在的设计,最后用了JSON来解决。
 
但是新的问题来了,对于put时是这样的:
/**
* 根据key设置map的值
*/
@Override
public V put(K key, V value) {
jedisTemp.hset(name, JSON.toJSONString(key), JSON.toJSONString(value));
return value;
}

这并没啥问题,因为对象转换成json串是正常的。问题是get的时候,我们使用的

alibaba.fastjson提供的接口并不能转回成具体类型的对象,因为get方法的的返回值是V类型,是泛型类型,没法得到class的type。
像这样的代码就不行啦:JSON.parseObject(json, V.class)。最后没办法,我只好把K和V的类型在创建时由调用者传入。看下面的代码里,两个红色的参数,当然这也没问题,毕竟调用者是知道类型的:
CacheManager.getOrCreateCache("permissionCache", String.class, RoleMenu.class);

最终get方法的实现就是这样:

@Override
public V get(Object key) {
String json = jedisTemp.hget(name, JSON.toJSONString(key));
return (V) JSON.parseObject(json, valueClass);
}

问题虽然是解决了,只不过总觉得怪怪的。

 
 
总结与反思
      整套的设计受openfire的集群设计影响比较大,我基本是借鉴过来的,目前来看还是挺不错,最近准备尝试Ignite,非常容易就接入了系统。
 
      只是openfire使用的是java实现的方案(Hazelcast/Coherence

),这些都是带Map结构的,并不会有我遇到的Redis的问题。但我觉得这套设计还挺不错,如果把map接口去掉,自己重新定义方法就可以解决这个问题,不使用泛型,当然这样对老代码的改动会比较大。
 
      还有一种情况就是多种缓存产品并存,比如同时使用redis和memcached,现有的设计可能支持不了。但是因为入口限制在了CacheManager,我想加个泛型支持就可以解决。只是这种场景或许并不多见吧。
 
 
 
注:此文章为原创,欢迎转载,请在文章页面明显位置给出此文链接!
若您觉得这篇文章还不错请点击下右下角的推荐,非常感谢!
http://www.cnblogs.com/5207

聊聊从web session的共享到可扩展缓存设计的更多相关文章

  1. session和cookie区别,多台WEB服务器如何共享session,禁用COOKIE后SESSION是否可用,为什么?

    答:session的运行机制: 用户A访问站点Y,如果站点Y指定了session_start();(以下假设session_start()总是存在)那么会产生一个session_id,这个sessio ...

  2. nginx+iis+redis+Task.MainForm构建分布式架构 之 (redis存储分布式共享的session及共享session运作流程)

    本次要分享的是利用windows+nginx+iis+redis+Task.MainForm组建分布式架构,上一篇分享文章制作是在windows上使用的nginx,一般正式发布的时候是在linux来配 ...

  3. ASP.NET Session的共享

    注: 在ashx文件中使用Session 首先添加引用 using System.Web.SessionState; 实现接口 public class XXXX: IHttpHandler ==&g ...

  4. 通过memcached来实现对tomcat集群中Session的共享策略

    近期在做一套集群的实现,实现的方案是在Linux下完成对Apache + Tomcat 负载均衡的功能. 上述功能已经实现,有需要了解的朋友可以看我另外一篇博文. Linux下Apache与Tomca ...

  5. Session分布式共享 = Session + Redis + Nginx

    一.Session 1.Session 介绍 我相信,搞Web开发的对Session一定再熟悉不过了,所以我就简单的介绍一下. Session:在计算机中,尤其是在网络应用中,称为"会话控制 ...

  6. 详解Session分布式共享(.NET CORE版)

    一.前言&回顾 在上篇文章Session分布式共享 = Session + Redis + Nginx中,好多同学留言问了我好多问题,其中印象深刻的有:nginx挂了怎么办?采用Redis的S ...

  7. spring-session实现分布式集群session的共享

    前言 HttpSession是通过Servlet容器创建和管理的,像Tomcat/Jetty都是保存在内存中的.但是我们把应用搭建成分布式的集群,然后利用LVS或Nginx做负载均衡,那么来自同一用户 ...

  8. Tomcat通过自带的Cluster方式实现Session会话共享环境操作记录

    一般来说,在多个tomcat集群业务中,session会话共享是必须的需求,不然前端nginx转发过来的请求不知道之前请求在哪台tomcat节点上,从而就找不到session以至于最终导致请求失败.要 ...

  9. redis缓存web session

    redis缓存web session 首先说下架构图.使用Redis作为会话服务器,统一管理Session.如图,集群里的WEB服务器共享存放在REDIS里面全部的客户端SESSION. 当然,反向代 ...

随机推荐

  1. CoreCRM 开发实录 —— Profile

    再简单的功能,也需要一坨代码的支持.Profile 的编辑功能主要就是修改个人的信息.比如用户名.头像.性别.电话--虽然只是一个编辑界面,但添加下来,涉及了6个文件的修改和7个新创建的文件.各种生成 ...

  2. SignalR代理对象异常:Uncaught TypeError: Cannot read property 'client' of undefined 推出的结论

    异常汇总:http://www.cnblogs.com/dunitian/p/4523006.html#signalR 后台创建了一个DntHub的集线器 前台在调用的时候出现了问题(经检查是代理对象 ...

  3. 执行 $Gulp 时发生了什么 —— 基于 Gulp 的前端集成解决方案(二)

    前言 文章 在windows下安装gulp —— 基于 Gulp 的前端集成解决方案(一) 中,已经完成对 gulp 的安装,由于是window环境,文中特意提到了可以通过安装 gitbash 来代替 ...

  4. 【NLP】Python NLTK处理原始文本

    Python NLTK 处理原始文本 作者:白宁超 2016年11月8日22:45:44 摘要:NLTK是由宾夕法尼亚大学计算机和信息科学使用python语言实现的一种自然语言工具包,其收集的大量公开 ...

  5. 基于Composer Player 模型加载和相关属性设置

    主要是基于达索软件Composer Player.的基础上做些二次开发. public class ComposerToolBarSetting { public bool AntiAliasingO ...

  6. SharePoint 2016 入门视频教程

    之前一直有朋友让自己录一些SharePoint的入门视频,之前没有太多时间,一个巧合的机会收到CSDN学院的邮件,可以在CSDN上发布视频教程,自己就录了一些.说起录视频也是蛮辛苦的,每天下班吃完饭要 ...

  7. Eclipse出现"Running Android Lint has encountered a problem"解决方案

    安装eclipse for android 时候的错误记录,转载自:http://blog.csdn.net/chenyufeng1991/article/details/47442555 (1)打开 ...

  8. android手机登录时遇到“QQ安全登录发现病毒”解决

    android手机作为开源系统非常容易感染病毒,有时候我们会经常遇到手机QQ登录时检测到app被感染,一般情况是由手机感染病毒所引起的,安装腾讯管家后只能检测病毒和卸载感染病毒的软件,不能清除病毒.解 ...

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

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

  10. [Django]用户权限学习系列之权限管理界面实现

    本系列前三章: http://www.cnblogs.com/CQ-LQJ/p/5604331.htmlPermission权限基本操作指令 http://www.cnblogs.com/CQ-LQJ ...