Spring ApplicationContext事件机制及使用

Spring中提供的标准事件:

  • ContextRefreshEvent,当ApplicationContext容器初始化完成或者被刷新的时候,就会发布该事件。比如调用ConfigurableApplicationContext接口中的refresh()方法。此处的容器初始化指的是所有的Bean都被成功装载,后处理(post-processor)Bean被检测到并且激活,所有单例Bean都被预实例化,ApplicationContext容器已经可以使用。只要上下文没有被关闭,刷新可以被多次触发。XMLWebApplicationContext支持热刷新,GenericApplicationContext不支持热刷新。

  • ContextStartedEvent,当ApplicationContext启动的时候发布事件,即调用ConfigurableApplicationContext接口的start方法的时候。这里的启动是指,所有的被容器管理生命周期的Bean接受到一个明确的启动信号。在经常需要停止后重新启动的场合比较适用。

  • ContextStoppedEvent,当ApplicationContext容器停止的时候发布事件,即调用ConfigurableApplicationContext的close方法的时候。这里的停止是指,所有被容器管理生命周期的Bean接到一个明确的停止信号。

  • ContextClosedEvent,当ApplicationContext关闭的时候发布事件,即调用ConfigurableApplicationContext的close方法的时候,关闭指的是所有的单例Bean都被销毁。关闭上下后,不能重新刷新或者重新启动。

  • RequestHandledEvent,只能用于DispatcherServlet的web应用,Spring处理用户请求结束后,系统会触发该事件。

注册到ZooKeeper

@Slf4j
@Component
public class RegisterZkListener implements ApplicationListener<ContextRefreshedEvent> {
    private ZkRegister zr;
    @Value("${zk.ip:}")
    public String zkHost;
    @Value("${register.zk.name}")
    public String registerZkName;
    @Value("${register.zk.port:8080}")
    public int registerZkPort;


    @Override
    public void onApplicationEvent(ContextRefreshedEvent event) {
        log.debug("onApplicationEvent is start");
        if (zr == null) {
            zr = new ZkRegister(zkHost);
        }
        zr.register(registerZkName, registerZkPort);
    }

}

缓存同步(自定义事件)

  • 自定义事件类CacheChangedEvent.java
@Getter
public class CacheChangedEvent extends ApplicationEvent {
    private final Set<Integer>  blogIds;
    public CacheChangedEvent(Object source, Set<Integer> blogIds) {
        super(source);
        this.blogIds = blogIds;
    }
}
  • 事件使用TenmaoFlowManager.java
public class TenmaoFlowManager implements DistCacheObserver, ApplicationListener<CacheChangedEvent> {
    enum CacheState {
        INITIALIZING,
        BROADCASTING
    }

    private volatile CacheState cacheState = CacheState.INITIALIZING;

    @Override
    public void onApplicationEvent(CacheChangedEvent event) {
        //使用两次校验,提高性能
        if (cacheState == CacheState.INITIALIZING) {
            synchronized (this) {
                if (cacheState == CacheState.INITIALIZING) {
                    log.info("add to cache events: {}", event.getblogIds());
                    lastEvent = event.getblogIds();
                    return;
                }
            }
        }
        doHandle(event.getblogIds());
    }
    private void doHandle(Set<Integer> blogIds) {
        //todo
    }
}
  • 事件发布DistCacheManager.java
public class DistCacheManager {
    private static final String CACHE_CHANNEL = "channel.blog.tenmao.cache";
    private static final String CACHE_KEY = "blog.tenmao.cache.tasks";

    static {
        ThreadFactory factory = new ThreadFactoryBuilder().setDaemon(true).setNameFormat("dist cache %d").build();
        THREAD_POOL_EXECUTOR = new ThreadPoolExecutor(2, 2, 1000, TimeUnit.MILLISECONDS, new ArrayBlockingQueue<>(100), factory);
    }

    @Resource
    private ApplicationContext applicationContext;
    private static final ThreadPoolExecutor THREAD_POOL_EXECUTOR;

    @Resource
    private JedisCluster jedis;

    private final JedisPubSub jedisPubSub = new JedisPubSub() {
        @Override
        public void onMessage(String channel, String message) {
            super.onMessage(channel, message);
            Set<Integer> blogIds = jedis.smembers(Tenmao_CACHE_KEY).stream().map(Integer::parseInt).collect(Collectors.toSet());
            log.info("get message: channel[{}], message[{}], blogIds[{}]", channel, message, blogIds);
            synchronized (this) {
                applicationContext.publishEvent(new CacheChangedEvent(this, blogIds));
            }
        }
    };
}

参考

©著作权归作者所有,转载或内容合作请联系作者
【社区内容提示】社区部分内容疑似由AI辅助生成,浏览时请结合常识与多方信息审慎甄别。
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

相关阅读更多精彩内容

  • 参考W3C Spring教程 Spring致力于J2EE应用的各种解决方案,而不仅仅专注于某一层解决方案。可以说S...
    王侦阅读 4,894评论 0 6
  • Spring容器高层视图 Spring 启动时读取应用程序提供的Bean配置信息,并在Spring容器中生成一份相...
    Theriseof阅读 7,855评论 1 24
  • 本来是准备看一看Spring源码的。然后在知乎上看到来一个帖子,说有一群**自己连Spring官方文档都没有完全读...
    此鱼不得水阅读 11,838评论 4 21
  • 文章较长,建议先收藏,在较空的时候来看。 Spring Boot 框架的初衷:快速地启动Spring应用Sprin...
    Drew_Zhong阅读 7,773评论 3 15
  • 蒸腾的热气迷惑了视线,徐徐缥缈的香氛醉了还带有凉气的鼻尖,急切的手指被滚烫极速弹开。 细浪舞动的清波,调皮的冲破丝...
    雪韵_莲心阅读 4,062评论 5 18

友情链接更多精彩内容