убить -9 пид???
kill может отправить заданное сообщение программе. Предустановленное сообщение SIGTERM(15) завершает указанную программу. Если программу по-прежнему нельзя завершить, можно попытаться принудительно удалить программу с помощью сообщения SIGKILL(9). Номер программы или задания можно просмотреть с помощью команды ps или команды jobs (это из учебника для новичков).
Чтобы усложнить задачу, проще говоря, она используется для уничтожения процессов в Linux, что? Что такое процесс, спросите вы меня? Пожалуйста, сделайте свой Baidu.
Я считаю, что многие люди использовали команду kill -9 pid, что означает полное уничтожение процесса, в целом мы используем ее без вышеперечисленных проблем, но в нашем проекте использование может иметь фатальные последствия. проблемы.
Проблемы, вызванные kill -9 pid
Так как kill -9 это грубое удаление, то оно может иметь более серьезные последствия для программы, так какие же именно последствия?
Как пример: функция перевода, а потом на два счета добавить деньги при списании внезапно отключили электричество? Что произойдет в это время? Для механизма хранения InnoDB ничего не теряется, потому что он поддерживает транзакции, но для механизма MyISAM это катастрофа, почему? Если вы списали деньги на счет А, а теперь вам нужно добавить деньги на счет Б, то при отключении электричества это приведет к тому, что деньги А были списаны, а Б не получил деньги, что категорически не допускается в производственной среде kill -9 эквивалентен эффекту внезапного сбоя питания.
Конечно, например, перевод денег, конечно, не с использованием движка MyISAM, но в настоящее время очень распространен распределенный огонь, межсервисный перевод, на этот раз, если вы используете kill -9 для остановки службы, это не ваша транзакция может гарантировать точность данных, на этот раз вы можете подумать о распределенных транзакциях, в этом мире нет абсолютной системы безопасности или архитектуры, распределенные транзакции также. В этом мире нет абсолютной системы безопасности или архитектуры, и распределенные транзакции одинаковы, они могут тоже есть проблемы, вероятность мала, а если и случится, то ущерб может оказаться непоправимым, поэтому нельзя использовать kill -9 для остановки службы, ведь вы не знаете, какие последствия он вызовет.
Это более очевиден в движке MyISAM, например, информация о пользователе хранится в двух таблицах, администратору необходимо изменить две таблицы при изменении информации о пользователе, но из-за вашей грубой силы kill -9 для завершения проекта, что приводит к успешной модификации только одна таблица, это тоже приведет к несоответствию данных, это мелочь, потому что большое дело в том, чтобы снова изменить, но деньги, контракты и другая важная информация, если из-за вашего перебора Удаление приведет к рассогласованию, я думаю, что это может будь серьезнее, чем удалить библиотеку и убежать, по крайней мере удаление можно восстановить, ты даже не знаешь, что в этом плохого.
Итак, как мы должны завершить проект?
其实java给我们提供了结束项目的功能,比如:tomcat可以使用shutdown.bat/shutdown.sh进行优雅结束。
什么叫优雅结束?
Шаг 1. Прекратите получать запросы и внутренние потоки.
Шаг 2. Определите, выполняется ли поток.
Шаг 3. Подождите, пока выполняющийся поток завершит выполнение.
Шаг 4. Остановите контейнер .
Вышеупомянутые четыре шага являются нормальным завершением процесса, так как же Springboot нормально завершает службу? Здесь я представляю несколько вариантов нормального завершения службы.
Окончание службы с элегантностью
убить -15 пид
Этот способ также завершит процесс (проект) более изящно, вам нужно быть осторожным при его использовании, почему? Давайте посмотрим на пример
Я написал обычный метод контроллера для тестирования:
@GetMapping(value = "/test") public String test(){ log.info("test --- start"); try { Thread.sleep(100000); } catch (InterruptedException e) { e.printStackTrace(); } log.info("test --- end"); return "test"; } 1234567891011
Код простой, после печати: test — start даем программе спящий режим на 100 секунд и потом печатаем: test — конец, в спящем потоке используем kill -15 pid для завершения процесса, угадайте какой тест — end будет напечатан?
application.yml
server: port: 9988 12
Запустить проект
sudo mvn spring-boot:run 1
Вот как maven запускает проект Springboot
Это означает, что проект запущен и работает.
Найдите идентификатор процесса проекта
sudo ps -ef |grep shutdown 1
Это номер процесса проекта, затем мы тестируем тестовый интерфейс, позволяем потоку заснуть, а затем останавливаем проект командой kill -15 14086.
sudo curl 127.0.0.1:9988/test 1
Вернуться к журналу проекта.
Обнаруживаем, что запрос дошел до сервиса и поток успешно ушел в спящий режим, теперь убиваем -15 14086 для завершения процесса
sudo kill -15 14086 1
Вернуться к журналу
2020-04-24 10:53:14.939 INFO 14086 --- [nio-9988-exec-1] com.ymy.controller.TestController : test --- start 2020-04-24 10:54:02.450 INFO 14086 --- [extShutdownHook] o.s.s.concurrent.ThreadPoolTaskExecutor : Shutting down ExecutorService 'applicationTaskExecutor' java.lang.InterruptedException: sleep interrupted at java.lang.Thread.sleep(Native Method) at com.ymy.controller.TestController.test(TestController.java:26) at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method) at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:62) at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43) at java.lang.reflect.Method.invoke(Method.java:498) at org.springframework.web.method.support.InvocableHandlerMethod.doInvoke(InvocableHandlerMethod.java:190) at org.springframework.web.method.support.InvocableHandlerMethod.invokeForRequest(InvocableHandlerMethod.java:138) at org.springframework.web.servlet.mvc.method.annotation.ServletInvocableHandlerMethod.invokeAndHandle(ServletInvocableHandlerMethod.java:105) at org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerAdapter.invokeHandlerMethod(RequestMappingHandlerAdapter.java:879) at org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerAdapter.handleInternal(RequestMappingHandlerAdapter.java:793) at org.springframework.web.servlet.mvc.method.AbstractHandlerMethodAdapter.handle(AbstractHandlerMethodAdapter.java:87) at org.springframework.web.servlet.DispatcherServlet.doDispatch(DispatcherServlet.java:1040) at org.springframework.web.servlet.DispatcherServlet.doService(DispatcherServlet.java:943) at org.springframework.web.servlet.FrameworkServlet.processRequest(FrameworkServlet.java:1006) at org.springframework.web.servlet.FrameworkServlet.doGet(FrameworkServlet.java:898) at javax.servlet.http.HttpServlet.service(HttpServlet.java:634) at org.springframework.web.servlet.FrameworkServlet.service(FrameworkServlet.java:883) at javax.servlet.http.HttpServlet.service(HttpServlet.java:741) at org.apache.catalina.core.ApplicationFilterChain.internalDoFilter(ApplicationFilterChain.java:231) at org.apache.catalina.core.ApplicationFilterChain.doFilter(ApplicationFilterChain.java:166) at org.apache.tomcat.websocket.server.WsFilter.doFilter(WsFilter.java:53) at org.apache.catalina.core.ApplicationFilterChain.internalDoFilter(ApplicationFilterChain.java:193) at org.apache.catalina.core.ApplicationFilterChain.doFilter(ApplicationFilterChain.java:166) at org.springframework.web.filter.RequestContextFilter.doFilterInternal(RequestContextFilter.java:100) at org.springframework.web.filter.OncePerRequestFilter.doFilter(OncePerRequestFilter.java:119) at org.apache.catalina.core.ApplicationFilterChain.internalDoFilter(ApplicationFilterChain.java:193) at org.apache.catalina.core.ApplicationFilterChain.doFilter(ApplicationFilterChain.java:166) at org.springframework.web.filter.FormContentFilter.doFilterInternal(FormContentFilter.java:93) at org.springframework.web.filter.OncePerRequestFilter.doFilter(OncePerRequestFilter.java:119) at org.apache.catalina.core.ApplicationFilterChain.internalDoFilter(ApplicationFilterChain.java:193) at org.apache.catalina.core.ApplicationFilterChain.doFilter(ApplicationFilterChain.java:166) at org.springframework.boot.actuate.metrics.web.servlet.WebMvcMetricsFilter.doFilterInternal(WebMvcMetricsFilter.java:109) at org.springframework.web.filter.OncePerRequestFilter.doFilter(OncePerRequestFilter.java:119) at org.apache.catalina.core.ApplicationFilterChain.internalDoFilter(ApplicationFilterChain.java:193) at org.apache.catalina.core.ApplicationFilterChain.doFilter(ApplicationFilterChain.java:166) at org.springframework.web.filter.CharacterEncodingFilter.doFilterInternal(CharacterEncodingFilter.java:201) at org.springframework.web.filter.OncePerRequestFilter.doFilter(OncePerRequestFilter.java:119) at org.apache.catalina.core.ApplicationFilterChain.internalDoFilter(ApplicationFilterChain.java:193) at org.apache.catalina.core.ApplicationFilterChain.doFilter(ApplicationFilterChain.java:166) at org.apache.catalina.core.StandardWrapperValve.invoke(StandardWrapperValve.java:202) at org.apache.catalina.core.StandardContextValve.invoke(StandardContextValve.java:96) at org.apache.catalina.authenticator.AuthenticatorBase.invoke(AuthenticatorBase.java:541) at org.apache.catalina.core.StandardHostValve.invoke(StandardHostValve.java:139) at org.apache.catalina.valves.ErrorReportValve.invoke(ErrorReportValve.java:92) at org.apache.catalina.core.StandardEngineValve.invoke(StandardEngineValve.java:74) at org.apache.catalina.connector.CoyoteAdapter.service(CoyoteAdapter.java:343) at org.apache.coyote.http11.Http11Processor.service(Http11Processor.java:373) at org.apache.coyote.AbstractProcessorLight.process(AbstractProcessorLight.java:65) at org.apache.coyote.AbstractProtocol$ConnectionHandler.process(AbstractProtocol.java:868) at org.apache.tomcat.util.net.NioEndpoint$SocketProcessor.doRun(NioEndpoint.java:1594) at org.apache.tomcat.util.net.SocketProcessorBase.run(SocketProcessorBase.java:49) at java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1149) at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:624) at org.apache.tomcat.util.threads.TaskThread$WrappingRunnable.run(TaskThread.java:61) at java.lang.Thread.run(Thread.java:748) 2020-04-24 10:54:04.574 INFO 14086 --- [nio-9988-exec-1] com.ymy.controller.TestController : test --- end 2020-04-24 10:54:04.610 ERROR 14086 --- [nio-9988-exec-1] o.s.web.servlet.HandlerExecutionChain : HandlerInterceptor.afterCompletion threw exception java.lang.NullPointerException: null at org.springframework.boot.actuate.metrics.web.servlet.LongTaskTimingHandlerInterceptor.stopLongTaskTimers(LongTaskTimingHandlerInterceptor.java:123) ~[spring-boot-actuator-2.2.6.RELEASE.jar:2.2.6.RELEASE] at org.springframework.boot.actuate.metrics.web.servlet.LongTaskTimingHandlerInterceptor.afterCompletion(LongTaskTimingHandlerInterceptor.java:79) ~[spring-boot-actuator-2.2.6.RELEASE.jar:2.2.6.RELEASE] at org.springframework.web.servlet.HandlerExecutionChain.triggerAfterCompletion(HandlerExecutionChain.java:179) ~[spring-webmvc-5.2.5.RELEASE.jar:5.2.5.RELEASE] at org.springframework.web.servlet.DispatcherServlet.triggerAfterCompletion(DispatcherServlet.java:1427) [spring-webmvc-5.2.5.RELEASE.jar:5.2.5.RELEASE] at org.springframework.web.servlet.DispatcherServlet.doDispatch(DispatcherServlet.java:1060) [spring-webmvc-5.2.5.RELEASE.jar:5.2.5.RELEASE] at org.springframework.web.servlet.DispatcherServlet.doService(DispatcherServlet.java:943) [spring-webmvc-5.2.5.RELEASE.jar:5.2.5.RELEASE] at org.springframework.web.servlet.FrameworkServlet.processRequest(FrameworkServlet.java:1006) [spring-webmvc-5.2.5.RELEASE.jar:5.2.5.RELEASE] at org.springframework.web.servlet.FrameworkServlet.doGet(FrameworkServlet.java:898) [spring-webmvc-5.2.5.RELEASE.jar:5.2.5.RELEASE] at javax.servlet.http.HttpServlet.service(HttpServlet.java:634) [tomcat-embed-core-9.0.33.jar:9.0.33] at org.springframework.web.servlet.FrameworkServlet.service(FrameworkServlet.java:883) [spring-webmvc-5.2.5.RELEASE.jar:5.2.5.RELEASE] at javax.servlet.http.HttpServlet.service(HttpServlet.java:741) [tomcat-embed-core-9.0.33.jar:9.0.33] at org.apache.catalina.core.ApplicationFilterChain.internalDoFilter(ApplicationFilterChain.java:231) [tomcat-embed-core-9.0.33.jar:9.0.33] at org.apache.catalina.core.ApplicationFilterChain.doFilter(ApplicationFilterChain.java:166) [tomcat-embed-core-9.0.33.jar:9.0.33] at org.apache.tomcat.websocket.server.WsFilter.doFilter(WsFilter.java:53) [tomcat-embed-websocket-9.0.33.jar:9.0.33] at org.apache.catalina.core.ApplicationFilterChain.internalDoFilter(ApplicationFilterChain.java:193) [tomcat-embed-core-9.0.33.jar:9.0.33] at org.apache.catalina.core.ApplicationFilterChain.doFilter(ApplicationFilterChain.java:166) [tomcat-embed-core-9.0.33.jar:9.0.33] at org.springframework.web.filter.RequestContextFilter.doFilterInternal(RequestContextFilter.java:100) [spring-web-5.2.5.RELEASE.jar:5.2.5.RELEASE] at org.springframework.web.filter.OncePerRequestFilter.doFilter(OncePerRequestFilter.java:119) [spring-web-5.2.5.RELEASE.jar:5.2.5.RELEASE] at org.apache.catalina.core.ApplicationFilterChain.internalDoFilter(ApplicationFilterChain.java:193) [tomcat-embed-core-9.0.33.jar:9.0.33] at org.apache.catalina.core.ApplicationFilterChain.doFilter(ApplicationFilterChain.java:166) [tomcat-embed-core-9.0.33.jar:9.0.33] at org.springframework.web.filter.FormContentFilter.doFilterInternal(FormContentFilter.java:93) [spring-web-5.2.5.RELEASE.jar:5.2.5.RELEASE] at org.springframework.web.filter.OncePerRequestFilter.doFilter(OncePerRequestFilter.java:119) [spring-web-5.2.5.RELEASE.jar:5.2.5.RELEASE] at org.apache.catalina.core.ApplicationFilterChain.internalDoFilter(ApplicationFilterChain.java:193) [tomcat-embed-core-9.0.33.jar:9.0.33] at org.apache.catalina.core.ApplicationFilterChain.doFilter(ApplicationFilterChain.java:166) [tomcat-embed-core-9.0.33.jar:9.0.33] at org.springframework.boot.actuate.metrics.web.servlet.WebMvcMetricsFilter.doFilterInternal(WebMvcMetricsFilter.java:109) [spring-boot-actuator-2.2.6.RELEASE.jar:2.2.6.RELEASE] at org.springframework.web.filter.OncePerRequestFilter.doFilter(OncePerRequestFilter.java:119) [spring-web-5.2.5.RELEASE.jar:5.2.5.RELEASE] at org.apache.catalina.core.ApplicationFilterChain.internalDoFilter(ApplicationFilterChain.java:193) [tomcat-embed-core-9.0.33.jar:9.0.33] at org.apache.catalina.core.ApplicationFilterChain.doFilter(ApplicationFilterChain.java:166) [tomcat-embed-core-9.0.33.jar:9.0.33] at org.springframework.web.filter.CharacterEncodingFilter.doFilterInternal(CharacterEncodingFilter.java:201) [spring-web-5.2.5.RELEASE.jar:5.2.5.RELEASE] at org.springframework.web.filter.OncePerRequestFilter.doFilter(OncePerRequestFilter.java:119) [spring-web-5.2.5.RELEASE.jar:5.2.5.RELEASE] at org.apache.catalina.core.ApplicationFilterChain.internalDoFilter(ApplicationFilterChain.java:193) [tomcat-embed-core-9.0.33.jar:9.0.33] at org.apache.catalina.core.ApplicationFilterChain.doFilter(ApplicationFilterChain.java:166) [tomcat-embed-core-9.0.33.jar:9.0.33] at org.apache.catalina.core.StandardWrapperValve.invoke(StandardWrapperValve.java:202) [tomcat-embed-core-9.0.33.jar:9.0.33] at org.apache.catalina.core.StandardContextValve.invoke(StandardContextValve.java:96) [tomcat-embed-core-9.0.33.jar:9.0.33] at org.apache.catalina.authenticator.AuthenticatorBase.invoke(AuthenticatorBase.java:541) [tomcat-embed-core-9.0.33.jar:9.0.33] at org.apache.catalina.core.StandardHostValve.invoke(StandardHostValve.java:139) [tomcat-embed-core-9.0.33.jar:9.0.33] at org.apache.catalina.valves.ErrorReportValve.invoke(ErrorReportValve.java:92) [tomcat-embed-core-9.0.33.jar:9.0.33] at org.apache.catalina.core.StandardEngineValve.invoke(StandardEngineValve.java:74) [tomcat-embed-core-9.0.33.jar:9.0.33] at org.apache.catalina.connector.CoyoteAdapter.service(CoyoteAdapter.java:343) [tomcat-embed-core-9.0.33.jar:9.0.33] at org.apache.coyote.http11.Http11Processor.service(Http11Processor.java:373) [tomcat-embed-core-9.0.33.jar:9.0.33] at org.apache.coyote.AbstractProcessorLight.process(AbstractProcessorLight.java:65) [tomcat-embed-core-9.0.33.jar:9.0.33] at org.apache.coyote.AbstractProtocol$ConnectionHandler.process(AbstractProtocol.java:868) [tomcat-embed-core-9.0.33.jar:9.0.33] at org.apache.tomcat.util.net.NioEndpoint$SocketProcessor.doRun(NioEndpoint.java:1594) [tomcat-embed-core-9.0.33.jar:9.0.33] at org.apache.tomcat.util.net.SocketProcessorBase.run(SocketProcessorBase.java:49) [tomcat-embed-core-9.0.33.jar:9.0.33] at java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1149) [na:1.8.0_242] at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:624) [na:1.8.0_242] at org.apache.tomcat.util.threads.TaskThread$WrappingRunnable.run(TaskThread.java:61) [tomcat-embed-core-9.0.33.jar:9.0.33] at java.lang.Thread.run(Thread.java:748) [na:1.8.0_242] 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112
Я не могу поверить, что он сообщает об ошибке, но тест — конец распечатывается, так почему же он сообщает об ошибке? Это связано с методом sleep, из-за которого sleep генерирует исключение, когда вызывается метод прерывания потока, когда поток находится в спящем режиме. Вот почему мы все еще можем видеть: test — end.ConfigurableApplicationContext colse
Давайте сначала посмотрим, как этого добиться
package com.ymy.controller; import lombok.extern.slf4j.Slf4j; import org.springframework.beans.BeansException; import org.springframework.context.ApplicationContext; import org.springframework.context.ApplicationContextAware; import org.springframework.context.ConfigurableApplicationContext; import org.springframework.web.bind.annotation.GetMapping; import org.springframework.web.bind.annotation.PostMapping; import org.springframework.web.bind.annotation.RestController; @RestController @Slf4j public class TestController implements ApplicationContextAware { private ApplicationContext context; @Override public void setApplicationContext(ApplicationContext applicationContext) throws BeansException { this.context = applicationContext; } @GetMapping(value = "/test") public String test(){ log.info("test --- start"); try { Thread.sleep(100000); } catch (InterruptedException e) { e.printStackTrace(); } log.info("test --- end"); return "test"; } /** * 停机 */ @PostMapping(value = "shutdown") public void shutdown(){ ConfigurableApplicationContext cyx = (ConfigurableApplicationContext) context; cyx.close(); } } 1234567891011121314151617181920212223242526272829303132333435363738394041424344
Сосредоточьтесь на: cyx.close();, почему он может остановить проект Springboot? Пожалуйста, смотрите исходный код.
public void close() { synchronized(this.startupShutdownMonitor) { this.doClose(); if (this.shutdownHook != null) { try { Runtime.getRuntime().removeShutdownHook(this.shutdownHook); } catch (IllegalStateException var4) { } } } } 123456789101112
Программа регистрирует ловушку отключения с помощью jvm при запуске, мы удалим эту ловушку отключения, когда выполним метод colse, и jvm будет знать, что это необходимо для остановки службы.
Посмотрим на результаты теста
По-видимому, он также отклонился от метода прерывания потока, заставив поток сообщить об ошибке, почти так же, как kill -15.
привод
Этот подход заключается в остановке службы путем введения зависимости. Актуатор предоставляет множество интерфейсов, таких как проверка работоспособности, основная информация и т. д. Мы также можем использовать его для изящной остановки.
Знакомство с зависимостями
<dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-actuator</artifactId> </dependency> 12345
application.yml
server: port: 9988 management: endpoints: web: exposure: include: shutdown endpoint: shutdown: enabled: true server: port: 8888 12345678910111213
Здесь я переделал интерфейс для привода, что повышает безопасность, так что давайте проверим его ниже.
@RequestMapping(value = "/test",method = RequestMethod.GET) public String test(){ System.out.println("test --- start"); try { Thread.sleep(10000); } catch (InterruptedException e) { e.printStackTrace(); } System.out.println("test --- end"); return "hello"; } 1234567891011
Остановка службы на пути к запросу теста.
Мы обнаруживаем, что отправка запроса на остановку службы также возвращает нам предупреждающее сообщение, очень удобное для пользователя, давайте посмотрим на консоль.
test — end выполняется, но метод прерывания потока по-прежнему вызывается при остановке пула потоков, в результате чего sleep сообщает об ошибке, все три этих способа могут более изящно остановить службу springboot, если в моем проекте есть потоки спать, я хочу остановить службу через 10 секунд, можно? Определенно можно, нам просто нужно сделать некоторые небольшие изменения, которые могут быть.
1. Новый класс службы остановки Springboot: ElegantShutdownConfig.java
package com.ymy.config; import org.apache.catalina.connector.Connector; import org.springframework.boot.web.embedded.tomcat.TomcatConnectorCustomizer; import org.springframework.context.ApplicationListener; import org.springframework.context.event.ContextClosedEvent; import java.util.concurrent.Executor; import java.util.concurrent.ThreadPoolExecutor; import java.util.concurrent.TimeUnit; public class ElegantShutdownConfig implements TomcatConnectorCustomizer, ApplicationListener<ContextClosedEvent> { private volatile Connector connector; private final int waitTime = 10; @Override public void customize(Connector connector) { this.connector = connector; } @Override public void onApplicationEvent(ContextClosedEvent event) { connector.pause(); Executor executor = connector.getProtocolHandler().getExecutor(); if (executor instanceof ThreadPoolExecutor) { try { ThreadPoolExecutor threadPoolExecutor = (ThreadPoolExecutor) executor; threadPoolExecutor.shutdown(); if (!threadPoolExecutor.awaitTermination(waitTime, TimeUnit.SECONDS)) { System.out.println("请尝试暴力关闭"); } } catch (InterruptedException ex) { System.out.println("异常了"); Thread.currentThread().interrupt(); } } } } 1234567891011121314151617181920212223242526272829303132333435363738394041
2. Добавьте bean-компонент в класс запуска
import com.ymy.config.ElegantShutdownConfig; import lombok.extern.slf4j.Slf4j; import org.apache.catalina.connector.Connector; import org.springframework.boot.SpringApplication; import org.springframework.boot.autoconfigure.SpringBootApplication; import org.springframework.boot.web.embedded.tomcat.TomcatConnectorCustomizer; import org.springframework.boot.web.embedded.tomcat.TomcatServletWebServerFactory; import org.springframework.boot.web.servlet.server.ServletWebServerFactory; import org.springframework.context.ApplicationListener; import org.springframework.context.ConfigurableApplicationContext; import org.springframework.context.annotation.Bean; import org.springframework.context.event.ContextClosedEvent; import java.util.concurrent.Executor; import java.util.concurrent.ThreadPoolExecutor; import java.util.concurrent.TimeUnit; @SpringBootApplication public class ShutdownServerApplication { public static void main(String[] args) { ConfigurableApplicationContext run = SpringApplication.run(ShutdownServerApplication.class, args); run.registerShutdownHook(); } @Bean public ElegantShutdownConfig elegantShutdownConfig() { return new ElegantShutdownConfig(); } @Bean public ServletWebServerFactory servletContainer() { TomcatServletWebServerFactory tomcat = new TomcatServletWebServerFactory(); tomcat.addConnectorCustomizers(elegantShutdownConfig()); return tomcat; } } 1234567891011121314151617181920212223242526272829303132333435363738394041
Таким образом, мы его настроили, давайте еще раз протестируем. Интерфейс теста все еще бездействует в течение 10 секунд.
Мы обнаружили, что на этот раз об ошибке не сообщается, он ждет некоторое время перед завершением пула потоков, на этот раз это время ожидания, которое мы настроили в классе ElegantShutdownConfig.
Итак, может быть, вам интересно, jvm не остановился сразу, так что же происходит, когда в это время есть запросы? Если во время завершения работы есть новый запрос, служба не получит этот запрос.
Операция резервного копирования данных
Если я хочу выполнить операцию резервного копирования, когда служба остановлена или что-то в этом роде, как мне это сделать? Очень просто добавить аннотацию к методу, который вы хотите выполнить: @PreDestroy
Так что вместе это означает, что он будет выполняться один раз перед остановкой контейнера, и вы можете делать в нем операции резервного копирования, или вы можете записывать время простоя и т. д.
Добавьте новый класс инструмента резервного копирования остановки службы: DataBackupConfig.javapackage com.ymy.config;
import org.springframework.context.annotation.Configuration; import javax.annotation.PreDestroy; @Configuration public class DataBackupConfig { @PreDestroy public void backData(){ System.out.println("正在备份数据。。。。。。。。。。。"); } } 123456789101112131415
Спасибо за прочтение. Если эта статья была вам полезна, ставьте лайк🧡
Буду продолжать усердно работать! 🧡
Если у вас есть вопросы по статье, пишите мне в личные сообщения🧡