`
wzucxd
  • 浏览: 26893 次
  • 性别: Icon_minigender_1
  • 来自: 杭州
文章分类
社区版块
存档分类
最新评论

加载远程服务上的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
分享到:
评论

相关推荐

    Spring Boot多模块配置文件读取

    - 当项目规模进一步扩大时,可以考虑使用Spring Cloud Config Server集中管理所有模块的配置,实现配置的远程加载和统一管理。 总之,Spring Boot的多模块配置文件读取策略旨在提高项目的可扩展性和可维护性。通过...

    springboot[1]-多模块共用配置文件.zip

    在Spring Boot项目中,多模块共用配置文件是一种常见的需求,尤其在大型复杂项目中,为了保持代码结构清晰,我们通常会将项目划分为多个独立的模块,如服务模块、数据访问模块、Web模块等。每个模块都有其特定的功能...

    spring-context-4.2.xsd.zip

    《Spring框架中的Context模块与XSD配置详解》 在Java企业级开发中,Spring框架扮演着至关重要的角色,尤其在Spring的Context模块中,通过XML Schema(XSD)进行配置,使得应用程序的组件管理和依赖注入变得简洁高效...

    计算机专业Java外文翻译 中文+英文(Spring框架介绍)

    在服务定位和配置管理上,Spring的ApplicationContext接口提供了加载配置信息和获取bean实例的能力。 Spring的模块化设计是其另一个重要特点,它由多个独立的子框架组成,包括核心容器(Core Container)、数据访问...

    Spring 框架简介

    - **Web支持**: Spring提供了强大的Web开发支持,包括MVC框架、远程调用、Web服务等功能。 #### 四、Spring框架的模块介绍 - **Spring Core**: 提供了框架的基本组成部分,包括IoC容器和依赖注入。 - **Spring AOP*...

    spring dubbo maven 整合demo

    2. 分模块开发:根据业务逻辑,将项目划分为服务提供者(provider)和服务消费者(consumer)两个子模块。 3. 引入依赖:在pom.xml文件中添加Spring、Dubbo和相关依赖。 4. 配置服务提供者:编写服务接口和实现类,...

    spring security 参考手册中文版

    远程处理 - spring-security-remoting.jar 25 Web - spring-security-web.jar 25 配置 - spring-security-config.jar 26 LDAP - spring-security-ldap.jar 26 ACL - spring-security-acl.jar 26 CAS - spring-...

    springboot多模块打包源码

    配置模块可以用来存放全局的配置文件,比如数据库连接、服务器端口等,通过Spring Cloud Config Server可以实现配置的统一管理和远程加载。 打包过程通常涉及到以下步骤: 1. **配置Maven**: 在每个模块的`pom....

    未来10年:OSGi、Spring_DM

    OSGi提供了一种强大的模块化体系结构,使得应用程序可以被拆分成独立的、可重用的组件(称为bundles),并能够在运行时动态加载、卸载、更新这些组件,从而极大地提高了系统的灵活性和可维护性。 **OSGi 4.2...

    Maven构建全栈式Flex、BlazeDS和Spring集成解决方案–第二部分_编写to-do_list服务器端

    按照指示,在`todolist`目录下执行`mvn archetype:create -DgroupId=org.epseelon.samples -DartifactId=todolist-web -DarchetypeArtifactId=maven-archetype-webapp`命令,这将生成一个新的子模块`todolist-web`。...

    Spring框架入门教程(新版)内含源码以及说明书可以自己运行复现.zip

    Spring框架的核心特性是控制反转(IoC)和面向切面编程(AOP),并提供了丰富的支持,如数据访问、事务管理、远程服务调用等。 **2. Spring IoC容器** Spring IoC(Inverse of Control,控制反转)容器是Spring的...

    springsurce

    Spring框架是Java领域中最重要的框架之一,它包含了许多模块,涵盖了Web应用开发的各个方面,如数据访问、事务管理、远程服务、Web服务等。 压缩包中的子文件名称列出了Spring框架的多个关键模块: 1. **spring-...

    research-spring-boot-2.4

    Spring Boot 2.4允许在每个模块的配置文件中定义特定于该模块的属性,并通过`@ConfigurationProperties`注解绑定到Java Bean上,使属性的使用和管理更加规范。 2. **共享属性**:尽管每个模块可以有自己的属性,但...

    Maven构建的分布式demo

    父POM可以作为所有子模块的模板,简化了配置的维护。 2. **WAR工程**:Web应用程序通常被打包成WAR(Web ARchive)文件。在分布式系统中,一个服务可能就是一个独立的WAR应用,部署在Web服务器上。通过在POM中设置...

    Spring Batch 参考文档

    - **Spring Batch Integration模块**:加强了与其他Spring技术(如Spring Integration)的集成能力。 - **升级到支持Spring 4和Java 8**:利用新版本的Spring框架和Java语言特性,提升了性能和开发体验。 - **...

    Java_SSH框架案例教程学生用书.pdf

    Spring框架还包括事务管理、远程调用等模块,极大地简化了Java EE应用程序的开发。 5. Ajax技术:Ajax(Asynchronous JavaScript and XML)是web2.0时代的一项重要技术,它允许在不重新加载整个网页的情况下,对...

    Apache Karaf调研文档

    OSGi 最重要的特性是在 Java 中 ClassLoader 是非常重要的概念,而大家也知道,JVM 本身在ClassLoader 上并没有提供非常强大的功能,比如模块开发非常重要的模块隔离 ClassLoader 的机制、版本加载机制等。...

    《深入理解OSGi:Equinox原理、应用与最佳实践》迷你书

    本书详细阐述了OSGi模块的建立、描述、依赖关系的处理,Bundle的启动原理和调度管理,以及如何处理本地和远程服务。 3. OSGi服务与Equinox应用实践:详细讲解了OSGi服务规范和企业级规范中最常用的几个子规范和技术...

    GWT 初级入门,中高级应用 开发文档手册 api(全)

    学习如何创建和管理子模块,以及如何使用.gwt.xml文件配置。理解编译选项,如代码分割,减少初始加载时间。了解GWT的本地化支持,包括多语言和日期时间格式。 六、GWT高级特性 1. RequestFactory:提供了一种更简洁...

Global site tag (gtag.js) - Google Analytics