阿里:别再手动部署jar包,动态上传热部署真香

近期开发系统过程中遇到的一个需求,系统给定一个接口,用户可以自定义开发该接口的实现,并将实现打成jar包,上传到系统中。系统完成热部署,并切换该接口的实现。

定义简单的接口

这里以一个简单的计算器功能为例,接口定义比较简单,直接上代码

public interface Calculator { int calculate(int a, int b); int add(int a, int b);}

该接口的一个简单的实现

考虑到用户实现接口的两种方式,使用spring上下文管理的方式,或者不依赖spring管理的方式,这里称它们为注解方式和反射方式。calculate方法对应注解方式,add方法对应反射方式。计算器接口实现类的代码如下:

@Servicepublic class CalculatorImpl implements Calculator { @Autowired CalculatorCore calculatorCore; /** * 注解方式 */ @Override public int calculate(int a, int b) { int c = calculatorCore.add(a, b); return c; } /** * 反射方式 */ @Override public int add(int a, int b) { return new CalculatorCore().add(a, b); }}

这里注入CalculatorCore的目的是为了验证在注解模式下,系统可以完整的构造出bean的依赖体系,并注册到当前spring容器中。CalculatorCore的代码如下:

@Servicepublic class CalculatorCore { public int add(int a, int b) { return a+b; }}

反射方式热部署

用户把jar包上传到系统的指定目录下,这里定义上传jar文件路径为jarAddress,jar的Url路径为jarPath。

private static String jarAddress = “E:/zzq/IDEA_WS/CalculatorTest/lib/Calculator.jar”;private static String jarPath = “file:/” + jarAddress;

并且可以要求用户填写jar包中接口实现类的完整类名。接下来系统要把上传的jar包加载到当前线程的类加载器中,然后通过完整类名,加载得到该实现的Class对象。然后反射调用即可,完整代码:

/** * 热加载Calculator接口的实现 反射方式 */public static void hotDeployWithReflect() throws Exception { URLClassLoader urlClassLoader = new URLClassLoader(new URL[]{new URL(jarPath)}, Thread.currentThread().getContextClassLoader()); Class clazz = urlClassLoader.loadClass(“com.nci.cetc15.calculator.impl.CalculatorImpl”); Calculator calculator = (Calculator) clazz.newInstance(); int result = calculator.add(1, 2); System.out.println(result);}

注解方式热部署

如果用户上传的jar包含了spring的上下文,那么就需要扫描jar包里的所有需要注入spring容器的bean,注册到当前系统的spring容器中。其实,这就是一个类似的热加载+动态注册的过程。

直接上代码:

/** * 加入jar包后 动态注册bean到spring容器,包括bean的依赖 */public static void hotDeployWithSpring() throws Exception { Set classNameSet = DeployUtils.readJarFile(jarAddress); URLClassLoader urlClassLoader = new URLClassLoader(new URL[]{new URL(jarPath)}, Thread.currentThread().getContextClassLoader()); for (String className : classNameSet) { Class clazz = urlClassLoader.loadClass(className); if (DeployUtils.isSpringBeanClass(clazz)) { BeanDefinitionBuilder beanDefinitionBuilder = BeanDefinitionBuilder.genericBeanDefinition(clazz); defaultListableBeanFactory.registerBeanDefinition(DeployUtils.transformName(className), beanDefinitionBuilder.getBeanDefinition()); } }}

在这个过程中,将jar加载到当前线程类加载器的过程和之前反射方式是一样的。然后扫描jar包下所有的类文件,获取到完整类名,并使用当前线程类加载器加载出该类名对应的class对象。判断该class对象是否带有spring的注解,如果包含,则将该对象注册到系统的spring容器中。

DeployUtils包含读取jar包所有类文件的方法、判断class对象是否包含sping注解的方法、获取注册对象对象名的方法。代码如下:

/** * 读取jar包中所有类文件 */public static Set readJarFile(String jarAddress) throws IOException { Set classNameSet = new HashSet(); JarFile jarFile = new JarFile(jarAddress); Enumeration entries = jarFile.entries();//遍历整个jar文件 while (entries.hasMoreElements()) { JarEntry jarEntry = entries.nextElement(); String name = jarEntry.getName(); if (name.endsWith(“.class”)) { String className = name.replace(“.class”, “”).replaceAll(“/”, “.”); classNameSet.add(className); } } return classNameSet;}/** * 方法描述 判断class对象是否带有spring的注解 */public static boolean isSpringBeanClass(Class cla) { if (cla == null) { return false; } //是否是接口 if (cla.isInterface()) { return false; } //是否是抽象类 if (Modifier.isAbstract(cla.getModifiers())) { return false; } if (cla.getAnnotation(Component.class) != null) { return true; } if (cla.getAnnotation(Repository.class) != null) { return true; } if (cla.getAnnotation(Service.class) != null) { return true; } return false;}/** * 类名首字母小写 作为spring容器beanMap的key */public static String transformName(String className) { String tmpstr = className.substring(className.lastIndexOf(“.”) + 1); return tmpstr.substring(0, 1).toLowerCase() + tmpstr.substring(1);}

删除jar时,需要同时删除spring容器中注册的bean

在jar包切换或删除时,需要将之前注册到spring容器的bean删除。spring容器的bean的删除操作和注册操作是相逆的过程,这里要注意使用同一个spring上下文。

代码如下:

/** * 删除jar包时 需要在spring容器删除注入 */public static void delete() throws Exception { Set classNameSet = DeployUtils.readJarFile(jarAddress); URLClassLoader urlClassLoader = new URLClassLoader(new URL[]{new URL(jarPath)}, Thread.currentThread().getContextClassLoader()); for (String className : classNameSet) { Class clazz = urlClassLoader.loadClass(className); if (DeployUtils.isSpringBeanClass(clazz)) { defaultListableBeanFactory.removeBeanDefinition(DeployUtils.transformName(className)); } }}

测试

测试类手动模拟用户上传jar的功能。测试函数写了个死循环,一开始没有找到jar会抛出异常,捕获该异常并睡眠10秒。这时候可以把jar手动放到指定的目录下。

代码如下:

ApplicationContext applicationContext = new ClassPathXmlApplicationContext(“applicationContext.xml”); DefaultListableBeanFactory defaultListableBeanFactory = (DefaultListableBeanFactory) applicationContext.getAutowireCapableBeanFactory(); while (true) { try { hotDeployWithReflect();// hotDeployWithSpring();// delete(); } catch (Exception e) { e.printStackTrace(); Thread.sleep(1000 * 10); } }

原文链接:https://mp.weixin.qq.com/s/-pddExHI4fj94JUEpNczhg

郑重声明:本文内容及图片均整理自互联网,不代表本站立场,版权归原作者所有,如有侵权请联系管理员(admin#wlmqw.com)删除。
(0)
用户投稿
上一篇 2022年6月16日
下一篇 2022年6月16日

相关推荐

  • 为什么他们可以成功,而你不可以

    1、不要等待机会,而要创造机会。成功不是将来才有的,而是从决定去做的那一刻起,持续累积而成。大多数人想要改造这个世界,但却罕有人想改造自己。当你感到悲哀痛苦时,最好是去学些什么东西…

    2022年6月30日
  • 皮肤暗黄怎么美白

    去黄提亮并不难,方法要对? 去黄提亮怎么做? 一白遮百丑,只要是个女的都不想变成黄脸婆,除非买昂贵的护肤品保养当然我们也可以通过”其他方法”来黄美白,找对方…

    2022年8月16日
  • 企业开发小程序分销商城的好处

    企业开发小程序分销商城的好处   在各大APP占领市场的时候,另一个更简便的应用顺应而生,那就是小程序,很多人可能不知道小程序到底是什么,小程序不同于APP需要下载安装,它是一种即…

    2022年8月19日
  • 女人都希望自己老的慢一点,四种抗衰老的方法,不妨学起来

    女人最不愿听到的两个字,应该就是“衰老”吧。虽然谁也不能阻挡衰老的脚步,但每每看着自己逐渐老去的容颜,难免会觉得伤感。与其和衰老做斗争,不如优雅的老去,接受生命的安排。 很多时候,…

    2022年6月1日
  • 选手机产生了困扰,iPhone13ProMax和三星S22UItra该怎么选择?

    我朋友,最近用了五年的iPhoneXsmax想换掉了,于是就产生了这样的困扰:iPhone手机的好坏,说实话,他自己心知肚明。现在唯一让他纠结的是,网上看了好多人在说三星s22UI…

    2022年6月24日
  • 学校餐厅 单位餐厅云端扫脸识别(刷卡)系统说明

    一、 产品参数: 安 卓 消 费 机 的 技 术 参 数 操作系统: Android 8.1 主控芯片: RK3288(四核Cortex-A17+GPU Mail-T764,Up …

    2022年6月15日
  • 电脑系统怎么重装win7(u盘做系统详细步骤)

    当使用的win7系统出现中毒,系统卡顿蓝屏死机等情况时,我们需要如何重装win7系统修复呢?其实可以通过装机软件实现,下面就以小白装机软件为例介绍一下详细操作步骤。 1.到小白系统…

    2022年5月6日
  • 小益,无处安放的“真智能”魅力

    智能门锁的出现,很大程度上提升了人们的生活体验。随着智能家居理念的爆火,尤其是对比传统门锁,智能门锁的安全系数更高,作为入门的第一道防线,许多用户表示都逃不过智能门锁的真香定律,纷…

    2022年8月13日
  • 缓解晒伤的7个方法,以下情况需要就医

    炎炎夏日,酷暑难耐,很多人出去游玩或者日常都没有防晒意识,因此也造就了晒伤这个夏季门诊最多见的情况。那么什么是晒伤?晒伤一般都有哪些症状?出现晒伤后该如何紧急处理?出现哪些情况必须…

    2022年8月24日
  • 买一只充电宝,配的上你的iPhone手机:iWALK 口袋宝到手体验

    充电宝我们都不陌生,特别是iPhone 用户,几乎都会随身携带一款充电宝吧……说起充电宝,这是一款两极分化特别严重的产品,要么就是极轻薄,薄、小轻,要么就是大容量,厚、大、沉。其实…

    2022年6月24日

联系我们

联系邮箱:admin#wlmqw.com
工作时间:周一至周五,10:30-18:30,节假日休息