一、什么是服务预热?
所谓服务预热,就是在服务启动完成到对外提供服务之前,针对特定场景提供一些初始化准备操作,比如线程池预热、缓存预热、数据库预热、web预热和jvm预热等等,需要注意的是,预热操作需要在应用真正对外提供服务之前完成,那么我们就可以基于框架的事件或者扩展点来完成这个操作,比如ContextRefreshedEvent
和ApplicationReadyEvent
事件,InitializingBean
和ApplicationContextAware
扩展点,以及@PostConstruct
和init
方法等等。
缓存预热
连接池预热
和数据库连接池一样,我们可以在应用启动时,根据需要初始化若干连接放入连接池,从而避免请求过来的时候再创建而影响性能,比如我们使用的是jedis客户端,那么我们在配置连接池的时候做如下改造。
//这里我们利用InitializingBean接口来实现,因为InitializingBean可以在对象被Spring创建并且成员变量全部注入后执行。
@Component
public class PreheatJedisPool extends JedisPool implements InitializingBean {
@Autowired
JedisPoolConfig jedisPoolConfig;
@Override
public void afterPropertiesSet() throws Exception {
for (int i = 0; i < this.jedisPoolConfig.getMinIdle(); i++) {
this.getResource();
}
}
}
系统启动时,尝试获取若干连接(这里取最小闲置),此时连接池为空,生成连接后会放入连接池,等请求进来的时候就不会再去创建连接了。
热点数据预热
在实际开发中,我们可以利用大数据统计用户访问的热点数据,在项目启动时将这些热点数据提前查询并保存到Redis中 ,这样就避免了首次访问热点数据流量打到数据库层。
@Component
@Slf4j
public class PreheatHotData implements ApplicationListener<ApplicationReadyEvent> {
@Autowired
RedisTemplate redisTemplate;
@Override
public void onApplicationEvent(ApplicationReadyEvent event) {
try {
//加载热点数据到缓存的业务代码
.........
} catch (Exception e) {
log.error("onApplicationEvent occur error;",e);
}
}
}