加载远程服务上的spring独立子模块

背景:

假设一个web工程有3个独立业务子模块,user,home,rpc,3个独立子模块也独立发布,子模块独立发布后可以及时reloadweb工程的业务功能,3个子模块可以在任何服务器上,也可以是3个不同的公司来提供,如下图

 

 

一般我们的工程会采用spring来管理bean,在这种情况下要稍微改造下。看上去和热部署也有点类似。

这次的知识点以这个背景为例来进行讲解。

 

首先看下这样做的一些好处:

1.业务模块的独立按需加载,可以加快web容器的启动,每次只需要启动时加载自己需要的内容

2.发布更新更快速,独立发布子模块影响局部的业务功能

 

原理:

1.我们先做一个子模块user.jar(在10.20.150.216共享里)

user.jar里面就一个spring_user.xml和User.java

-------------------

spring_user.xml

<bean id="user"class="com.wzucxd.User"> 

    <property name="name"value="xxx" /> 

</bean>

-------------------

User.java

packagecom.wzucxd;

 

public class User {

 

    private Integer id;

   

    private String name;

 

    public Integer getId() {

        return id;

    }

 

    public void setId(Integer id) {

        this.id = id;

    }

 

    public String getName() {

        return name;

    }

 

    public void setName(String name) {

        this.name = name;

    }  

}

 

2.下面就是模拟父容器加载子模块的case

基本思路:通过XmlBeanFactory先加载子模块的bean对象的BeanDefinition,然后将BeanDefinition注册到父容易的beanFactory,最后对加载bean的AppClassloader改造成需要的ClassLoader(这里改造成URLClassLoader)

 

详细代码以及示例如下

 

public class Test {

 

    public static void main(String[] args) {

        try

             

        //读取spring全局配置文件,(这里是一个空内容的文件没有任何内容),即我们平时web project父容器上下文

        //当然平时这里会用ClassPathXmlApplicationContext看文件所在位置了。。。

            ApplicationContextapplicationContext = newFileSystemXmlApplicationContext("file://10.20.150.216\\share\\ebook\\j2se\\classloader\\spring_config.xml"); 

 

            //创建全局spring BeanFactory目的是将所有子模块的bean对象注册到这个父容器上下文中

            DefaultListableBeanFactorybeanFactory = (DefaultListableBeanFactory)applicationContext.getAutowireCapableBeanFactory(); 

 

            //独立模块的spring bean配置文件位置

            String configurationFilePath = "jar:file://10.20.150.216\\share\\ebook\\j2se\\classloader/user.jar!/spring_user.xml"

 

            //这里可以做一个逻辑如果该配置文件不存在那么父容器启动的时候这个子模块就不加载

            URL url = new URL(configurationFilePath); 

 

            //建立远程资源访问

            UrlResource urlResource = new UrlResource(url); 

 

            XmlBeanFactory xmlBeanFactory = new XmlBeanFactory(urlResource); 

 

            String[] beanIds =xmlBeanFactory.getBeanDefinitionNames(); 

 

            for (String beanId : beanIds) { 

                //获得的子模块bean对象

                BeanDefinition bd =xmlBeanFactory.getMergedBeanDefinition(beanId); 

                //在这里将子模块bean对象注册到父容易上下文中完成bean对象的Definition

               beanFactory.registerBeanDefinition(beanId, bd); 

            } 

 

            //接着要进行classloader的改变加载的class文件现在不在classpath而是其他地方远程共享、http服务或者其他协议服务的机器上

            //这时候就需要将父容易中beanFactory的加载beanclassloader改变父容易中beanFactory默认是AppClassLoader这种情况下改成用URLClassLoader

            // 以下这行设置BeanFactoryClassLoaderURLClassLoader以加载外部类 

            setBeanClassLoader(beanFactory); 

 

            //以下是测试是否注入成功

            //从父容器上下文中获取user对象

            Object pluginBean =applicationContext.getBean("user"); 

 

            //测试结果

            String val = tryInvoke(pluginBean); 

            System.out.println(val);

 

        } catch (Exception exc) { 

            exc.printStackTrace(); 

        } 

    }

   

    private static void setBeanClassLoader( 

            DefaultListableBeanFactorybeanFactory) 

            throws MalformedURLException { 

        //指明spring_user.xml配置出现的bean对象所在jar位置

        String jarFilePath = "file://10.20.150.216\\share\\ebook\\j2se\\classloader\\user.jar"

        URL jarUrl = new URL(jarFilePath);

        URL[] urls = new URL[] { jarUrl }; 

        URLClassLoader cl = new URLClassLoader(urls); 

       beanFactory.setBeanClassLoader(cl); 

    } 

 

    private static String tryInvoke(Object bean) throws SecurityException, 

            NoSuchMethodException,IllegalArgumentException, 

            IllegalAccessException,InvocationTargetException { 

 

        Class<?> paramTypes[] = new Class[0]; 

        Method method =bean.getClass().getDeclaredMethod("getName", paramTypes); 

        Object paramValues[] = new Object[0]; 

        Object obj = method.invoke(bean, paramValues);

        //.....

        return (String)obj;

    } 

}

 

这里如果我们不改变BeanClassLoader会有什么问题呢?会出现classnofound异常,这个主要原因就是默认的beanfactory classloader是AppClassLoader。

 

3.当然还有另外一种方式,将jar中的类扫描出来,自己创建个classloader,一个个添加进去,这种方式更加灵活多变。其实还是使用spring的bean管理方式使用比较方便,已经解决了主要场景。

    public static void main(String[] args) throws Exception {

        URL url = new URL("jar:file:d:\\user.jar!/"); 

       URLClassLoader uc = new URLClassLoader(new URL[]{url}); 

       Class<?> cls = uc.loadClass("com.wzucxd.User"); 

       Object obj = cls.newInstance(); 

       System.out.println(obj);

    }

 

 

最后大家可能会发现,与其说加载远程的独立spring子模块,其实其核心关键还是classloder这个知识的应用扩展, 关于classloader的详细内容可以参考以前整的classloader.doc
  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值