一种简单的本地cache的实现
我们经常在web应用中需要对某些数据进行本地cache,特别是一些存在数据库或者其他需要访问其他服务的数据进行本地cache,如果这些数据更新不频繁,或者很少更新,而且对数据更新允许有一定的时间才反映到线上的话,那么下面这种方案个人觉得比较简单方便,不需要使用memcache等集中式缓存之类来实现。
直接在web应用中使用定时器和InitializingBean来实现即可:
请看service类:
下面这个idNameCacheMap就是我们需要进行数据缓存的对象,在spring加载完之后的afterPropertiesSet获取缓存数据进行加载,然后使用pring的quartz定时器进行定时更新,
这种方式简单,而且不用担心集群中不同服务器数据的不一致,定时器更新能够保证数据的不一致性的时间非常短,而且应用已启动就会加载数据,定时器更新的频率更新需要进行设置。我看到有很多使用timer类来处理,在java代码中启动一个线程,线程启动后就加载数据,然后sleep一段时间后更新缓存数据。这种其实应用的不同服务器启动的时间相差比较大,可能导致服务器间数据不一致时间会比较长。
import java.util.HashMap; import java.util.Map; import javax.annotation.Resource; public class LocalCacheService implements InitializingBean{ private Map<Long, String> idNameCacheMap = new HashMap<Long, String>(); @Resource private DataService dataService; public void update(){ Map<Long,String> tmpMap = dataService.getLastest(); if(tmpMap != null && !tmpMap.isEmpty()){ this.idNameCacheMap = tmpMap; } } @Override public void afterPropertiesSet() throws Exception { try { update(); } catch (Exception e) { } } }
<bean id="localCacheService" class="org.LocalCacheService" > </bean> <bean id="localCacheListenerTrigger" class="org.springframework.scheduling.quartz.CronTriggerBean"> <property name="jobDetail"> <bean class="org.springframework.scheduling.quartz.MethodInvokingJobDetailFactoryBean"> <property name="targetObject" ref=""localCacheService""> </property> <property name="targetMethod"> <value>update</value> </property> </bean> </property> <property name="cronExpression"> <value>0 0/30 * * * ?</value> </property> </bean> <bean class="org.springframework.scheduling.quartz.SchedulerFactoryBean" autowire="no"> <property name="triggers"> <list> <ref bean="localCacheListenerTrigger" /> </list> </property> </bean>