在使用springboot進行開發的過程當中,咱們常常須要處理這樣的場景:在服務啓動的時候,須要向服務註冊中心(例如zk)註冊服務狀態,以便當服務狀態改變的時候,能夠故障摘除和負載均衡。web
我遇到過兩種註冊的途徑:spring
一、在Spring的webapplication啓動完成後,直接進行註冊;springboot
二、在servlet容器啓動完成後,經過listener進行註冊。app
本文經過一個demo講述一下這兩種註冊方式,使用的是傳統的向zk註冊的方案。負載均衡
一、Spring webapplication啓動完成後註冊webapp
先上代碼看一下ide
@SpringBootApplication public class WebApplication { private static final Logger logger = LoggerFactory.getLogger(WebApplication.class); private static volatile boolean IS_REGISTRY = false; public static void main(String[] args) { ApplicationContext context = run(WebApplication.class, args); if (IS_REGISTRY) { logger.info("註冊2: WebApplication啓動完成後"); ZkClient zkClient = context.getBean(ZkClient.class); zkClient.register(); IS_REGISTRY = true; logger.info("註冊2: 註冊成功"); } } }
這裏,咱們在WebApplication中,獲取zkClient,並進行註冊。this
這裏須要說明一點,咱們這裏經過ApplicationContext來獲取zkClient的bean,緣由是在webApplication的初始化過程當中你不能用Autowired的方式注入Bean,由於在webApplication啓動過程當中纔會讀全部的configuration並將bean初始化完成,在沒有完成初始化以前,你不能注入bean。spa
關於註冊的詳細代碼這裏不展開了。code
二、在servlet容器初始化完成後,經過listener的方式進行註冊
照樣先上代碼
@WebListener public class RegisterListener implements ServletContextListener { protected final Logger logger = LoggerFactory.getLogger(this.getClass()); private static volatile boolean IS_REGISTRY = false; @Autowired private ZkClient zkClient; @Override public void contextInitialized(ServletContextEvent servletContextEvent) { try { if (!IS_REGISTRY) { logger.info("註冊1: Servelet容器啓動成功後"); zkClient.register(); logger.info("註冊1: 註冊成功"); } IS_REGISTRY = true; } catch (Exception e) { IS_REGISTRY = false; logger.info("註冊1: 註冊失敗"); } } @Override public void contextDestroyed(ServletContextEvent servletContextEvent) { if (IS_REGISTRY) { zkClient.stop(); } } }
你須要先寫一個listener,這個listener實現ServletContextListener接口,而且用@WebListener進行註解,這是springboot註解式的listener書寫方式。
在servlet容器啓動成功以後,會調用這個監聽器的contextInitialized方法,servlet容器若是一旦銷燬,不能提供服務了,會調用監聽器的contextDestroyed方法。換句話說,這個監聽器在監聽servlet容器的狀態。
而後你只須要在application主類中打開listener配置就好。
@ServletComponentScan @SpringBootApplication public class WebApplication { }
三、這兩種方式的比較
對於一個對外提供http協議的web服務,在語義上servlet容器的註冊會顯得清晰一些,可是若是你的spring容器啓動時間過長的話,可能出現servlet初始化完成,而且已經註冊,可是服務不能對外提供訪問的gap time,因此我通常仍是使用第一種方式進行註冊。
這種場景是這樣的
能夠看到,當servlet註冊成功以後,其實webapplication尚未啓動完成,這個時候服務是不能正常提供訪問的。
在zk上能夠看到,兩次註冊都已經成功了。