背景:
假设一个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的加载bean的classloader改变(父容易中beanFactory默认是AppClassLoader,这种情况下改成用URLClassLoader)
// 以下这行设置BeanFactory的ClassLoader为URLClassLoader,以加载外部类
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 Cloud Config Server集中管理所有模块的配置,实现配置的远程加载和统一管理。 总之,Spring Boot的多模块配置文件读取策略旨在提高项目的可扩展性和可维护性。通过...
在Spring Boot项目中,多模块共用配置文件是一种常见的需求,尤其在大型复杂项目中,为了保持代码结构清晰,我们通常会将项目划分为多个独立的模块,如服务模块、数据访问模块、Web模块等。每个模块都有其特定的功能...
《Spring框架中的Context模块与XSD配置详解》 在Java企业级开发中,Spring框架扮演着至关重要的角色,尤其在Spring的Context模块中,通过XML Schema(XSD)进行配置,使得应用程序的组件管理和依赖注入变得简洁高效...
在服务定位和配置管理上,Spring的ApplicationContext接口提供了加载配置信息和获取bean实例的能力。 Spring的模块化设计是其另一个重要特点,它由多个独立的子框架组成,包括核心容器(Core Container)、数据访问...
- **Web支持**: Spring提供了强大的Web开发支持,包括MVC框架、远程调用、Web服务等功能。 #### 四、Spring框架的模块介绍 - **Spring Core**: 提供了框架的基本组成部分,包括IoC容器和依赖注入。 - **Spring AOP*...
2. 分模块开发:根据业务逻辑,将项目划分为服务提供者(provider)和服务消费者(consumer)两个子模块。 3. 引入依赖:在pom.xml文件中添加Spring、Dubbo和相关依赖。 4. 配置服务提供者:编写服务接口和实现类,...
远程处理 - 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-...
配置模块可以用来存放全局的配置文件,比如数据库连接、服务器端口等,通过Spring Cloud Config Server可以实现配置的统一管理和远程加载。 打包过程通常涉及到以下步骤: 1. **配置Maven**: 在每个模块的`pom....
OSGi提供了一种强大的模块化体系结构,使得应用程序可以被拆分成独立的、可重用的组件(称为bundles),并能够在运行时动态加载、卸载、更新这些组件,从而极大地提高了系统的灵活性和可维护性。 **OSGi 4.2...
按照指示,在`todolist`目录下执行`mvn archetype:create -DgroupId=org.epseelon.samples -DartifactId=todolist-web -DarchetypeArtifactId=maven-archetype-webapp`命令,这将生成一个新的子模块`todolist-web`。...
Spring框架的核心特性是控制反转(IoC)和面向切面编程(AOP),并提供了丰富的支持,如数据访问、事务管理、远程服务调用等。 **2. Spring IoC容器** Spring IoC(Inverse of Control,控制反转)容器是Spring的...
Spring框架是Java领域中最重要的框架之一,它包含了许多模块,涵盖了Web应用开发的各个方面,如数据访问、事务管理、远程服务、Web服务等。 压缩包中的子文件名称列出了Spring框架的多个关键模块: 1. **spring-...
Spring Boot 2.4允许在每个模块的配置文件中定义特定于该模块的属性,并通过`@ConfigurationProperties`注解绑定到Java Bean上,使属性的使用和管理更加规范。 2. **共享属性**:尽管每个模块可以有自己的属性,但...
父POM可以作为所有子模块的模板,简化了配置的维护。 2. **WAR工程**:Web应用程序通常被打包成WAR(Web ARchive)文件。在分布式系统中,一个服务可能就是一个独立的WAR应用,部署在Web服务器上。通过在POM中设置...
- **Spring Batch Integration模块**:加强了与其他Spring技术(如Spring Integration)的集成能力。 - **升级到支持Spring 4和Java 8**:利用新版本的Spring框架和Java语言特性,提升了性能和开发体验。 - **...
Spring框架还包括事务管理、远程调用等模块,极大地简化了Java EE应用程序的开发。 5. Ajax技术:Ajax(Asynchronous JavaScript and XML)是web2.0时代的一项重要技术,它允许在不重新加载整个网页的情况下,对...
OSGi 最重要的特性是在 Java 中 ClassLoader 是非常重要的概念,而大家也知道,JVM 本身在ClassLoader 上并没有提供非常强大的功能,比如模块开发非常重要的模块隔离 ClassLoader 的机制、版本加载机制等。...
本书详细阐述了OSGi模块的建立、描述、依赖关系的处理,Bundle的启动原理和调度管理,以及如何处理本地和远程服务。 3. OSGi服务与Equinox应用实践:详细讲解了OSGi服务规范和企业级规范中最常用的几个子规范和技术...
学习如何创建和管理子模块,以及如何使用.gwt.xml文件配置。理解编译选项,如代码分割,减少初始加载时间。了解GWT的本地化支持,包括多语言和日期时间格式。 六、GWT高级特性 1. RequestFactory:提供了一种更简洁...