服务预热问题

一、什么是服务预热?

所谓服务预热,就是在服务启动完成到对外提供服务之前,针对特定场景提供一些初始化准备操作,比如线程池预热、缓存预热、数据库预热、web预热和jvm预热等等,需要注意的是,预热操作需要在应用真正对外提供服务之前完成,那么我们就可以基于框架的事件或者扩展点来完成这个操作,比如ContextRefreshedEventApplicationReadyEvent事件,InitializingBeanApplicationContextAware扩展点,以及@PostConstructinit方法等等。


缓存预热

连接池预热

和数据库连接池一样,我们可以在应用启动时,根据需要初始化若干连接放入连接池,从而避免请求过来的时候再创建而影响性能,比如我们使用的是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);
        }
    }
}

你可能感兴趣的:(架构,架构)