日本综合一区二区|亚洲中文天堂综合|日韩欧美自拍一区|男女精品天堂一区|欧美自拍第6页亚洲成人精品一区|亚洲黄色天堂一区二区成人|超碰91偷拍第一页|日韩av夜夜嗨中文字幕|久久蜜综合视频官网|精美人妻一区二区三区

RELATEED CONSULTING
相關(guān)咨詢
選擇下列產(chǎn)品馬上在線溝通
服務(wù)時間:8:30-17:00
你可能遇到了下面的問題
關(guān)閉右側(cè)工具欄

新聞中心

這里有您想知道的互聯(lián)網(wǎng)營銷解決方案
SpringBoot如何熱加載Jar實現(xiàn)動態(tài)插件?

本文轉(zhuǎn)載自微信公眾號「陶陶技術(shù)筆記」,作者zlt2000。轉(zhuǎn)載本文請聯(lián)系陶陶技術(shù)筆記公眾號。

創(chuàng)新互聯(lián)建站堅持“要么做到,要么別承諾”的工作理念,服務(wù)領(lǐng)域包括:做網(wǎng)站、成都網(wǎng)站設(shè)計、企業(yè)官網(wǎng)、英文網(wǎng)站、手機端網(wǎng)站、網(wǎng)站推廣等服務(wù),滿足客戶于互聯(lián)網(wǎng)時代的豐都網(wǎng)站設(shè)計、移動媒體設(shè)計的需求,幫助企業(yè)找到有效的互聯(lián)網(wǎng)解決方案。努力成為您成熟可靠的網(wǎng)絡(luò)建設(shè)合作伙伴!

一、背景

動態(tài)插件化編程是一件很酷的事情,能實現(xiàn)業(yè)務(wù)功能的 「解耦」 便于維護(hù),另外也可以提升 「可擴展性」 隨時可以在不停服務(wù)器的情況下擴展功能,也具有非常好的 「開放性」 除了自己的研發(fā)人員可以開發(fā)功能之外,也能接納第三方開發(fā)商按照規(guī)范開發(fā)的插件。

常見的動態(tài)插件的實現(xiàn)方式有 SPI、OSGI 等方案,由于脫離了 Spring IOC 的管理在插件中無法注入主程序的 Bean 對象,例如主程序中已經(jīng)集成了 Redis 但是在插件中無法使用。

本文主要介紹在 Spring Boot 工程中熱加載 jar 包并注冊成為 Bean 對象的一種實現(xiàn)思路,在動態(tài)擴展功能的同時支持在插件中注入主程序的 Bean 實現(xiàn)功能更強大的插件。

二、熱加載 jar 包

通過指定的鏈接或者路徑動態(tài)加載 jar 包,可以使用 URLClassLoader 的 addURL 方法來實現(xiàn),樣例代碼如下:

「ClassLoaderUtil 類」

 
 
 
  1. public class ClassLoaderUtil {
  2.     public static ClassLoader getClassLoader(String url) {
  3.         try {
  4.             Method method = URLClassLoader.class.getDeclaredMethod("addURL", URL.class);
  5.             if (!method.isAccessible()) {
  6.                 method.setAccessible(true);
  7.             }
  8.             URLClassLoader classLoader = new URLClassLoader(new URL[]{}, ClassLoader.getSystemClassLoader());
  9.             method.invoke(classLoader, new URL(url));
  10.             return classLoader;
  11.         } catch (Exception e) {
  12.             log.error("getClassLoader-error", e);
  13.             return null;
  14.         }
  15.     }
  16. }

其中在創(chuàng)建 URLClassLoader 時,指定當(dāng)前系統(tǒng)的 ClassLoader 為父類加載器 ClassLoader.getSystemClassLoader() 這步比較關(guān)鍵,用于打通主程序與插件之間的 ClassLoader ,解決把插件注冊進(jìn) IOC 時的各種 ClassNotFoundException 問題。

三、動態(tài)注冊 Bean

將插件 jar 中加載的實現(xiàn)類注冊到 Spring 的 IOC 中,同時也會將 IOC 中已有的 Bean 注入進(jìn)插件中;分別在程序啟動時和運行時兩種場景下的實現(xiàn)方式。

3.1. 啟動時注冊

使用 ImportBeanDefinitionRegistrar 實現(xiàn)在 Spring Boot 啟動時動態(tài)注冊插件的 Bean,樣例代碼如下:「PluginImportBeanDefinitionRegistrar 類」

 
 
 
  1. public class PluginImportBeanDefinitionRegistrar implements ImportBeanDefinitionRegistrar {
  2.     private final String targetUrl = "file:/D:/SpringBootPluginTest/plugins/plugin-impl-0.0.1-SNAPSHOT.jar";
  3.     private final String pluginClass = "com.plugin.impl.PluginImpl";
  4.     @SneakyThrows
  5.     @Override
  6.     public void registerBeanDefinitions(AnnotationMetadata importingClassMetadata, BeanDefinitionRegistry registry) {
  7.         ClassLoader classLoader = ClassLoaderUtil.getClassLoader(targetUrl);
  8.         Class clazz = classLoader.loadClass(pluginClass);
  9.         BeanDefinitionBuilder builder = BeanDefinitionBuilder.genericBeanDefinition(clazz);
  10.         BeanDefinition beanDefinition = builder.getBeanDefinition();
  11.         registry.registerBeanDefinition(clazz.getName(), beanDefinition);
  12.     }
  13. }

3.2. 運行時注冊

程序運行時動態(tài)注冊插件的 Bean 通過使用 ApplicationContext 對象來實現(xiàn),樣例代碼如下:

 
 
 
  1. @GetMapping("/reload")
  2. public Object reload() throws ClassNotFoundException {
  3.   ClassLoader classLoader = ClassLoaderUtil.getClassLoader(targetUrl);
  4.   Class clazz = classLoader.loadClass(pluginClass);
  5.   springUtil.registerBean(clazz.getName(), clazz);
  6.   PluginInterface plugin = (PluginInterface)springUtil.getBean(clazz.getName());
  7.   return plugin.sayHello("test reload");
  8. }

「SpringUtil 類」

 
 
 
  1. @Component
  2. public class SpringUtil implements ApplicationContextAware {
  3.     private DefaultListableBeanFactory defaultListableBeanFactory;
  4.     private ApplicationContext applicationContext;
  5.     @Override
  6.     public void setApplicationContext(ApplicationContext applicationContext) throws BeansException {
  7.         this.applicationContext = applicationContext;
  8.         ConfigurableApplicationContext configurableApplicationContext = (ConfigurableApplicationContext) applicationContext;
  9.         this.defaultListableBeanFactory = (DefaultListableBeanFactory) configurableApplicationContext.getBeanFactory();
  10.     }
  11.     public void registerBean(String beanName, Class clazz) {
  12.         BeanDefinitionBuilder beanDefinitionBuilder = BeanDefinitionBuilder.genericBeanDefinition(clazz);
  13.         defaultListableBeanFactory.registerBeanDefinition(beanName, beanDefinitionBuilder.getRawBeanDefinition());
  14.     }
  15.     public Object getBean(String name) {
  16.         return applicationContext.getBean(name);
  17.     }
  18. }

四、總結(jié)

本文介紹的插件化實現(xiàn)思路通過 「共用 ClassLoader」 和 「動態(tài)注冊 Bean」 的方式,打通了插件與主程序之間的類加載器和 Spring 容器,使得可以非常方便的實現(xiàn)插件與插件之間和插件與主程序之間的 「類交互」,例如在插件中注入主程序的 Redis、DataSource、調(diào)用遠(yuǎn)程 Dubbo 接口等等。

但是由于沒有對插件之間的 ClassLoader 進(jìn)行 「隔離」 也可能會存在如類沖突、版本沖突等問題;并且由于 ClassLoader 中的 Class 對象無法銷毀,所以除非修改類名或者類路徑,不然插件中已加載到 ClassLoader 的類是沒辦法動態(tài)修改的。

所以本方案比較適合插件數(shù)據(jù)量不會太多、具有較好的開發(fā)規(guī)范、插件經(jīng)過測試后才能上線或發(fā)布的場景。

五、完整 demo

https://github.com/zlt2000/springs-boot-plugin-test


分享名稱:SpringBoot如何熱加載Jar實現(xiàn)動態(tài)插件?
標(biāo)題路徑:http://www.dlmjj.cn/article/djsecss.html