范文健康探索娱乐情感热点
投稿投诉
热点动态
科技财经
情感日志
励志美文
娱乐时尚
游戏搞笑
探索旅游
历史星座
健康养生
美丽育儿
范文作文
教案论文
国学影视

Spring容器关闭执行销毁方法有几种?

  什么是Spring的扩展点?
  这个问题让我很深刻,记得之前有一个面试就被问到有没有使用过。
  那他是什么?
  先来看下Spring容器的加载过程
  可以看到Bean从无到有主要是经历了四个步骤
  就是在成熟态的时候,在初始化生命周期执行回调方法
  主要是以接口或者注解的形式对外提供,注入到IOC容器中,完成对应的功能。哪些场景下,我们需要使用退出前销毁
  主要是希望在销毁之前在做一些事情,比如像池化技术正确的断开,JVM内存回收,还有业务逻辑执行。业务场景
  直接进入正题,我先说一说我的业务场景,在执行任务A的时候,这时候服务重启了,因为任务A加了分布式锁,所以在
  重启服务的时候,补偿机制拿到了任务A发现锁依然被占用着,所以我就希望能够在应用关闭之前把锁给释放掉,减少
  对补偿机制的影响。
  补充:这里其实也可以用Redisson,来进行锁续期,一段时间过后自己释放,但是系统中更多时候使用简单的分布式
  锁就可以满足,避免引入Redisson这么重的框架。解决方案将当前执行任务的redis锁记录下来在Spring应用关系的时候,调用销毁方法进行锁的释放采用SmartLifecycle和DisposableBean相互配合来执行destroy()方法
  具体实现:@Service public class UserServiceImpl implements UserService, DisposableBean, SmartLifecycle {      private volatile boolean running = false;      private List lockKeys = new ArrayList<>();      @Resource     HelloService helloService;      @Override     public void get() {         String key = "redis:key";         //伪代码         RedissonUtil.lock(key);         try {             lockKeys.add(key);         } catch (Exception ex){             ex.printStackTrace();             //...         } finally {             RedissonUtil.unlock(key);             lockKeys.remove(key);         }              }      @Override     public void destroy() {         // 删除正在执行中的key         RedissonUtil.deletes(lockKeys);         running = false;     }      @Override     public void start() {         System.out.println("start >>>>");         running = true;     }      @Override     public void stop() {         System.out.println("stop >>>>");         // 删除正在执行中的key         RedissonUtil.deletes(lockKeys);     }      @Override     public boolean isRunning() {         return running;     }      @Override     public int getPhase() {         return Integer.MAX_VALUE;     } } 复制代码
  利用DisposableBean和SmartLifecycle进行双重的销毁机制,如果已经执行了DisposableBean的销毁方法
  那可以修改running的值为false,就不会再进行stop()的执行了Spring执行关闭的时机JVM关闭对象销毁时候容器停止关闭前执行销毁方法有哪些DisposableBean
  调用时机:Bean对象销毁的时候@Service public class UserServiceImpl implements UserService, DisposableBean {      @Override     public void destroy() {         System.out.println("destroy>>>>>");     } } 复制代码SmartLifecycle
  调用时机:Spring容器发出关闭通知@Service public class UserServiceImpl implements UserService, SmartLifecycle {      @Override     public void start() {         System.out.println("start >>>>");         running = true;     }      @Override     public void stop() {         System.out.println("stop >>>>");     }      @Override     public boolean isRunning() {         return running;     }      @Override     public int getPhase() {         return Integer.MAX_VALUE;     } } 复制代码InitializingBean
  这个方式比较特殊,就是在初始化的时候,提前设置好了钩子函数addShutdownHook
  调用时机:监听到JVM关闭@Service public class UserServiceImpl implements UserService, InitializingBean {      @Override     public void afterPropertiesSet() {         Runtime.getRuntime().addShutdownHook(new Thread(() -> {             helloService.get();             System.out.println("addShutdownHook>>>>>");         }));     } } 复制代码@PreDestroy注解@PreDestroy public void preDestroy(){     System.out.println("PreDestroy>>>>"); } 复制代码Xml和@Bean绑定destoryMethod方法
  对比执行结果:
  SmartLifecycle > @PreDestroy,DisposableBean > addShutdownHook2022-09-05 23:06:04.046  INFO 11807 --- [           main] c.l.d.SpringBootDemoDockerApplication    : Started SpringBootDemoDockerApplication in 1.4 seconds (JVM running for 1.752) ApplicationRunner>>>>> CommandLineRunner>>>项目启动完毕后,倒数10秒关闭 thread1... thread1... thread1... thread1... stop >>>> 2022-09-05 23:06:14.054  INFO 11807 --- [           main] o.s.s.concurrent.ThreadPoolTaskExecutor  : Shutting down ExecutorService "applicationTaskExecutor" PreDestroy>>>> destroy>>>>> thread1... thread1... get addShutdownHook>>>>> 复制代码SmartLifecycle接口源码
  了解一下SmartLifecycle接口到底由哪些组成的/**  * 当上下文被刷新(所有对象已被实例化和初始化之后)时,将调用该方法  * isAutoStartup默认为true则调用start,否则需要自己手动调用  */ @Override public void start() {     System.out.println("start >>>>");     running = true; }  /**  * 接口Lifecycle子类的方法,只有非SmartLifecycle的子类才会执行该方法。  * 1. 该方法只对直接实现接口Lifecycle的类才起作用,对实现SmartLifecycle接口的类无效。  * 2. 方法stop()和方法stop(Runnable callback)的区别只在于,后者是SmartLifecycle子类的专属。  */ @Override public void stop() {     System.out.println("stop >>>>"); }  /**  * 只有该方法返回false时,start方法才会被执行  * 只有该方法返回true时,stop(Runnable callback)或stop()方法才会被执  * @return  */ @Override public boolean isRunning() {     return running; }  /**  * 返回 Integer.MAX_VALUE 仅表明  * 我们将是第一个关闭的 bean 和最后一个启动的 bean  * 关闭容器的第一时间调用stop()方法  */ @Override public int getPhase() {     return Integer.MAX_VALUE; }  /**  * 如果该`Lifecycle`类所在的上下文在调用`refresh`时,希望能够自己自动进行回调,则返回`true`,  * false的值表明组件打算通过显式的start()调用来启动,类似于普通的Lifecycle实现。  */ @Override public boolean isAutoStartup() {     return false; }  /**  * SmartLifecycle子类的才有的方法,当isRunning方法返回true时,该方法才会被调用。  * 很多框架中的源码中,都会把真正逻辑写在stop()方法内。  */  @Override public void stop(Runnable callback) {     stop();      // 如果你让isRunning返回true,需要执行stop这个方法     // 在程序退出时,Spring的DefaultLifecycleProcessor会认为这个MySmartLifecycle没有stop完成,     // 程序会一直卡着结束不了,等待一定时间(默认超时时间30秒)后才会自动结束。     callback.run(); } 复制代码
  SmartLifecycle#isRunning判断是否已经执行,false表示还未执行
  则调用SmartLifecycle#start()执行
  当关闭的时候isRunning为ture已经执行
  则调用SmartLifecycle#stop()执行学习MQ如何进行退出前优雅执行销毁方法
  DefaultRocketMQListenerContainer.classpublic class DefaultRocketMQListenerContainer implements InitializingBean,     RocketMQListenerContainer, SmartLifecycle, ApplicationContextAware {     private final static Logger log = LoggerFactory.getLogger(DefaultRocketMQListenerContainer.class);      private boolean running;      ...           @Override     public void destroy() {         this.setRunning(false);         if (Objects.nonNull(consumer)) {             consumer.shutdown();         }         log.info("container destroyed, {}", this.toString());     }      @Override     public boolean isAutoStartup() {         return true;     }      @Override     public void stop(Runnable callback) {         stop();         callback.run();     }      @Override     public void start() {         if (this.isRunning()) {             throw new IllegalStateException("container already running. " + this.toString());         }          try {             consumer.start();         } catch (MQClientException e) {             throw new IllegalStateException("Failed to start RocketMQ push consumer", e);         }         this.setRunning(true);          log.info("running container: {}", this.toString());     }      @Override     public void stop() {         if (this.isRunning()) {             if (Objects.nonNull(consumer)) {                 consumer.shutdown();             }             setRunning(false);         }     }      @Override     public boolean isRunning() {         return running;     }      private void setRunning(boolean running) {         this.running = running;     }      @Override     public int getPhase() {         // Returning Integer.MAX_VALUE only suggests that         // we will be the first bean to shutdown and last bean to start         return Integer.MAX_VALUE;     }      @Override     public void afterPropertiesSet() throws Exception {         initRocketMQPushConsumer();          this.messageType = getMessageType();         this.methodParameter = getMethodParameter();         log.debug("RocketMQ messageType: {}", messageType);     } } 复制代码
  RocketMQ在这里进行了几个步骤需要我们关注他将getPhase的值设置为最大,在容器关闭的第一时间调用stop()方法同时实现了SmartLifecycle和RocketMQListenerContainer接口,分别实现了stop()和destroy()方法, 进行双重关闭,如果和destroy()先执行了,则将running设置为false,不再执行stop()
  原文链接:https://juejin.cn/post/7139920679683489823

贵州,一个拼凑起来的省份,不完整的地理单元制约了经济发展贵州,位于中国西南地区。贵州在中国各省区中最特别的一点就是是中国唯一一个不沿大江大河不沿边不沿海的省份。贵州地处于云贵高原东部地区,但是却未能将云贵高原东部全部纳入版图之内,使得贵在沅陵,遇见最美的秋天沅陵,沈从文看一年也不会讨厌。沅陵,林则徐诗云一县好山留客住,五溪秋水为君清。他们被沅陵的山水风光人文风情所倾倒,为沅陵而歌,为沅陵而吟,把发自内心的由衷赞美,留在沅陵的土地上。沅南桑威奇群岛英国在南大西洋的海外领地,狂风骇浪中的冰火岛有这样一座群岛,它附近的海面上漂浮着冰山和冰渍碎块,但它自己却流淌着火山熔岩,来过这里的人寥寥无几,但它却是南极海洋动物的天堂上世纪阿根廷与英国在为此发生激烈的争夺战争,如今却并无清风破暑入画来来源人民网人民日报海外版浙江省宁波市四明山风景如画,吸引大批游客进山游玩。董小飞摄(人民图片)藤蔓似的盘山公路将我们送进云端,车子似一叶轻舟,荡漾在浙江省宁波市四明山八百里碧波之上这个上海迪士尼乐园独有的在建主题园区近期有一波剧透迪士尼粉丝大会上的疯狂动物城展区上海迪士尼乐园度假区微信公众号图上海迪士尼乐园正在建设中的第八个主题园区疯狂动物城,近期传来新的动向。自2019年上海迪士尼度假区宣布全球首个疯狂动焦山行宫西出大同三十里到云冈石窟,再往西行三十里到焦山。焦山乃小山包,周匝也就四五里,但形状奇特,如古代车轮横陈于武州川北岸。过桥依山势叠次而上,遇有庙塔和石窟。庙是泰山庙河神庙和白衣观音贵阳的第一豪宅白宫,成了网红打卡地,花了27亿打造的有多美?引导语贵州的省会贵阳以其独特的文化特色优势吸引了大批国内外游客前来欣赏。这其中有一个独特的景观名为白宫,说起白宫,或许会有人想到会不会是那个国际上非常知名的地方,它确实是以西欧风格良辰美景共此时文旅夜经济推动消费规模持续扩大夜间文化消费纵深化精细化多元化的发展,不仅为消费者带来丰富的文化消费体验,还带动了周边消费新业态。首批120个国家级夜间文化和旅游消费集聚区,通过积极培育夜间消费新业态新场景新热点建文帝隐匿在磁器口?明朝初年朱棣随着靖难之役的胜利荣登大宝,其侄子建文帝在明皇城一场大火后不知去向,朱棣多番派人寻找,连郑和下西洋据说也是去海外搜索建文帝下落。中华大地亦有多处有建文帝隐匿之处的传说,十一假期来武当山,来逍遥谷辟谷!来武当山做一次修仙之旅倾尽一生,我们都在寻找。寻找心中的净土与安宁,让喧闹的心回归于青山绿水之间。十一假期月来武当山,来逍遥谷,听琴音,感受武当魅力的太极气息!从繁华中抽身,我们相约一起去旅行,去心中的抗美援朝胜利在即,毛主席仍秘调24万志愿军赴朝,背后有何隐情?文怪兽学府编辑怪兽学府1953年6月,抗美援朝即将进入尾声,毛主席却突然派遣24万志愿军奔赴朝鲜前线,就连志愿军总司令彭德怀也感到很疑惑,就是这一决定让我们不得不佩服伟人的智慧和魄
生活究竟的规律的好,还是不要太规律太阳系绕着银心运动,这是规律。太阳系绕着银心运动的轨道在不停地改变,改变的是无规律的。为什么无规律?速度与方向都有发生着改变。为什么会这样?因为太阳自身的质量在变化,因为银河系自身秋收冬藏,补肾填精正当时,一个天地同补方,收阳气,藏肾精大家好,我是刘医生。中医经典黄帝内经里有一句话叫做春生夏长,秋收冬藏,这是对我们养生思路的指导。春天和夏天应该顺应温暖的气候,温补阳气,到了秋天和冬天,我们就应该像自然界的动物一样冬季最易被寒湿侵入,记住这2招,排出全身寒湿,把阳气藏得足足的润燥是贯穿整个秋冬的主题,但正因如此,我们往往忽略了寒湿。眼下这个时节,气温骤降风露加重,外界的阴寒愈发凝重。如果你最近莫名精神萎靡全身疲惫不堪四肢乏力,甚至腰酸背痛,各个关节,总书虫双语听读4级下极限之旅0305前情提要1977年9月9日娜奥米出发了,航行的第六天塞乐马特的舵坏了,她试着修理的时候,舵掉进了海里,她花了几个小时换了一个新的舵,第12天她在那利群岛补给了一个塞乐马特的舵,她经海南日记走进黎乡作者沈念(湖南省作协副主席,第八届鲁迅文学奖散文奖得主)海南岛的山是海的皱褶。深秋的海南,我走在琼中的山道上。去海南的人多为看海,而我此行却是转山。前往琼中和五指山,是海南热带雨林自驾三日游滁州安徽人游安徽三日游方案1自驾线路明光滁州定远凤阳游览景点黄寨草场醉翁亭琅琊寺琅琊阁红石峡明中都鼓楼狼巷迷谷小岗村大包干纪念馆行程安排第一天各地明光市黄寨草场滁州市酒店第二天早上酒店琅琊山游览中这届红毯恩怨多!新仇旧怨同台,baby不理文咏珊,林允对上陈妍希终于有届热闹的红毯了!金鸡奖闭幕,33位女星参加晚会,走了红毯。各家铆足了劲头打扮,争奇斗艳,想要拔得头筹。娱乐圈毕竟是个圈,红毯上人一多,新仇旧怨也就聚齐了。红毯比美也就势在必得老来得子,李玉成新欢小腹疑似怀孕,纹身引争议,是真的吗?老来得子,李玉成新欢小腹隆起疑似怀孕,腿上纹身引争议,是真的吗?马玉琴跟李玉成这对夫妻,想必大家都不陌生。女方85岁男方53岁,是一对老少恋,近年来一直感情稳定。但是因为马玉琴将两李易峰嫖娼被抓后,张一山被曝大瓜这次,他一点也不冤李易峰塌房事件后,娱乐圈可谓是闷雷不断。前阵子,有网友爆料,张一山被抓了,北京朋友说的。近年来,倒在朝阳群众举报下的明星数不胜数。北京朋友确实容易让人相信。消息一出,迅速引起了网友冯小刚徐帆移民被锤?曝美移民局负责人证实8月份便通过审核为何娱乐圈大牌明星都会在成名后纷纷移民,且小心翼翼不敢被人知晓?自十月底冯小刚导演被拍到在美国豪宅深夜送客画面后,网上便频繁出现他移民美国的新闻,引发众网友关注。据悉,早在2015李少莉再次公开现身!官媒发文,批其发布会不是秀场一,新闻发布会不是秀场李少莉副局长因为佩戴奢侈品,以及用手指读发言稿火上热搜。还被一些网友调侃为丝巾姐大波浪姐等。都说高手在民间,在李少莉火出圈以后,她在发布会上的形象,还被某些人