使用Spring Session做分布式会话管理

  在Web项目开发中,会话管理是一个很重要的部分,用于存储与用户相关的数据。通常是由符合session规范的容器来负责存储管理,也就是一旦容器关闭,重启会导致会话失效。因此打造一个高可用性的系统,必须将session管理从容器中独立出来。而这实现方案有很多种,下面简单介绍下:

  第一种是使用容器扩展来实现,大家比较容易接受的是通过容器插件来实现,比如基于Tomcat的tomcat-redis-session-manager,基于Jetty的jetty-session-redis等等。好处是对项目来说是透明的,无需改动代码。不过前者目前还不支持Tomcat 8,或者说不太完善。个人觉得由于过于依赖容器,一旦容器升级或者更换意味着又得从新来过。并且代码不在项目中,对开发者来说维护也是个问题。

  第二种是自己写一套会话管理的工具类,包括Session管理和Cookie管理,在需要使用会话的时候都从自己的工具类中获取,而工具类后端存储可以放到Redis中。很显然这个方案灵活性最大,但开发需要一些额外的时间。并且系统中存在两套Session方案,很容易弄错而导致取不到数据。

  第三种是使用框架的会话管理工具,也就是本文要说的spring-session,可以理解是替换了Servlet那一套会话管理,既不依赖容器,又不需要改动代码,并且是用了spring-data-redis那一套连接池,可以说是最完美的解决方案。当然,前提是项目要使用Spring Framework才行。

  这里简单记录下整合的过程:

  如果项目之前没有整合过spring-data-redis的话,这一步需要先做,在maven中添加这两个依赖:

<dependency>
	<groupId>org.springframework.data</groupId>
	<artifactId>spring-data-redis</artifactId>
	<version>1.5.2.RELEASE</version>
</dependency>
<dependency>
	<groupId>org.springframework.session</groupId>
	<artifactId>spring-session</artifactId>
	<version>1.0.2.RELEASE</version>
</dependency>

  再在applicationContext.xml中添加以下bean,用于定义redis的连接池和初始化redis模版操作类,自行替换其中的相关变量。

<!-- redis -->
<bean id="jedisPoolConfig" class="redis.clients.jedis.JedisPoolConfig">
</bean>

<bean id="jedisConnectionFactory"
	class="org.springframework.data.redis.connection.jedis.JedisConnectionFactory">
	<property name="hostName" value="${redis.host}" />
	<property name="port" value="${redis.port}" />
	<property name="password" value="${redis.pass}" />
	<property name="timeout" value="${redis.timeout}" />
	<property name="poolConfig" ref="jedisPoolConfig" />
	<property name="usePool" value="true" />
</bean>

<bean id="redisTemplate" class="org.springframework.data.redis.core.StringRedisTemplate">
	<property name="connectionFactory" ref="jedisConnectionFactory" />
</bean>

<!-- 将session放入redis -->
<bean id="redisHttpSessionConfiguration"
class="org.springframework.session.data.redis.config.annotation.web.http.RedisHttpSessionConfiguration">
	<property name="maxInactiveIntervalInSeconds" value="1800" />
</bean>

  这里前面几个bean都是操作redis时候使用的,最后一个bean才是spring-session需要用到的,其中的id可以不写或者保持不变,这也是一个约定优先配置的体现。这个bean中又会自动产生多个bean,用于相关操作,极大的简化了我们的配置项。其中有个比较重要的是springSessionRepositoryFilter,它将在下面的代理filter中被调用到。maxInactiveIntervalInSeconds表示超时时间,默认是1800秒。写上述配置的时候我个人习惯采用xml来定义,官方文档中有采用注解来声明一个配置类。

  然后是在web.xml中添加一个session代理filter,通过这个filter来包装Servlet的getSession()。需要注意的是这个filter需要放在所有filter链最前面

<!-- delegatingFilterProxy -->
<filter>
	<filter-name>springSessionRepositoryFilter</filter-name>
	<filter-class>org.springframework.web.filter.DelegatingFilterProxy</filter-class>
</filter>
<filter-mapping>
	<filter-name>springSessionRepositoryFilter</filter-name>
	<url-pattern>/*</url-pattern>
</filter-mapping>

  这样便配置完毕了,需要注意的是,spring-session要求Redis Server版本不低于2.8

  验证:使用redis-cli就可以查看到session key了,且浏览器Cookie中的jsessionid已经替换为session。

127.0.0.1:6379> KEYS *
1) "spring:session:expirations:1440922740000"
2) "spring:session:sessions:35b48cb4-62f8-440c-afac-9c7e3cfe98d3"

闪光灯那些事(二)

  自从走上玩灯的道路后,便一发不可收拾,俗话说人像玩灯,风景玩镜,而我更喜欢结合两者,烈日下依然可以压低天空亮度。同样喜欢用灯的朋友可以看看传说中的“一灯大师”Zack Arias的闪灯视频。

  由于刚入门的时候选择了一款小巧的Nissin i40,但该灯不支持主控,以及功率略低。仅作机顶使用到没啥问题,但要离机引闪就不行了,首先不支持手动,毕竟人家是需要转盘来控制的。其次闪光补偿幅度也很小,可能是本身功率就很小的缘故吧。于是打算换一支主控灯,首先当然是考虑国产灯了,便宜嘛。

  在闪客日记论坛中泡了几天后入了一支沃龙SP-600,此灯在国产货中口碑算是不错的,做工也还凑活,唯一不足的是在我永诺622c离机引闪下曝光过度,大概误差有2ev样子,但机顶使用是正常的,升级了622c-tx依旧如此,没办法只好退了。之后又购入了斯丹德DF-800,这货拿到手就感觉low到极致了,做工不是一般的粗糙,连扩散板都很难抠出来,打出来的光斑也是不规则,暗角超级大,感觉虚标了闪光指数,只好退货处理。两只灯都有一个共同的问题:连续全光输出几次后电池发热极其严重,取出来都烫手。灯头倒是不怎么热,我可是在空调房里面测试的,用的是爱乐普电池,真怕爆炸。。。至于永诺的灯,不敢再试了,网评烧的最多的就是它了,可人家销量高啊,大概都抱着坏了再换的打算而用的。

  最后还是选择了佳能自己的600ex-rt,做工啥的都不用说了,使用622c完美引闪,不愧为原厂最强灯,测试下来耗电也不算大,关键是电池也没见发烫,安全第一,太棒了。

  写下这篇文章纪录下折腾的一些心得,想在摄影这条路一直走下去的朋友,一步到位还是很有道理的,能省下不少时间,毕竟时间才是最宝贵的。