我们经常在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>