weblogic有一个很贴心的功能,允许把多个war应用共同依赖的jar包,打包一个单独的war,以libary方式部署,然后各应用在weblogic.xml里声明引用该libary即可,这样可大大减少打包后的war文件尺寸,可以加快部署的上传进度,对web server而言,这类共用jar包只加载一次,也节省资源。
但是jboss下想达到类似的功能就要复杂很多了,先来一个简单的示例:
一、基础篇
1.1 假如我们先开发了一个工具库,打包后生成的mylib.jar ,为了减少各种依赖项的干扰,这个工程不依赖任何其它第三方库,只有一个测试方法:
package yjmyzz.test;public class TestUtils { public String sayHello(String msg) { return "Hello ," + msg + " !"; }}
1.2 再建一个spring mvc的web项目,为了简单起见,也只有一个空页面,啥功能也没有,pom.xml里引用刚才的mylib工程
12 yjmyzz 3mylib 41.0 5compile 6
Controller里,调用mylib中的sayHello方法
@RequestMapping("/index")public String index() { TestUtils utils = new TestUtils(); System.out.println(utils.sayHello("hello")); return "index";}
然后打包,默认情况下mylib.jar会打进myweb项目的WEB-INF/lib下
前面这些都是准备工作而已,不用太在意具体细节,下面才是正经开始.
现在我们要把这个mylib-1.0.jar给弄到jboss里,让它成为jboss的默认模块,最终目的是myweb.war的WEB-INF/lib目录下,不再需要这个jar。
1.3 折腾 jboss EAP
a. $JBOSS_HOME/modules 目录下,创建 mylib/main 目录,即要保证$JBOSS_HOME/modules/mylib/main目录存在
b. 把mylib-1.0.jar复制到$JBOSS_HOME/modules/mylib/main中
c. 在$JBOSS_HOME/modules/mylib/main中创建一个名为module.xml的文件,内容如下:
1 23 4 65
注意:第2行中的name="xxx"这里的name一定要跟/modules/下的子目录名一致,如果子目录是多层结构,比如 /modules/mycompany/mylib/main,则这里的name应该是"mycompany.mylib"
1.4 应用新增加的module
module加好以后,myweb在启动时并不知道jboss新增了这个module,有3种方法可以处理:
a) 修改myweb.war中META-INF/MANIFEST.MF清单文件
Manifest-Version: 1.0Dependencies: mylibBuilt-By: jimmyBuild-Jdk: 1.7.0_79Created-By: Apache Maven 3.3.3Archiver-Version: Plexus Archiver
注意第2行, Dependencies: mylib 把这行加上后,就表示myweb.war在启动时,要依赖mylib这个module,如果我们自己开发的module不止一个,多个module之间要英文逗号隔开,比如:module1,module2
当然,这一行如果要人工手动添加,未免太不讲究,有maven-plugin可以帮我们搞定:
12 org.apache.maven.plugins 3maven-war-plugin 45 116 107 9mylib 8
b) 还有第2种方式,在WEB-INF下添加文件jboss-deployment-structure.xml,内容如下:
12 3 74 65
c) 上面二种方式,都是由应用本身来控制加载哪些模块,也是我个人推荐的方式,还有一种一劳永逸的暴力方法,直接修改$JBOSS_HOME/standalone/configuration/standalone.xml文件(注:如果是domain模式,则相应修改domain.xml文件),内容如下:
12 3 54 false 6true 7
根据关键字":ee:1.1"找到这段, 添加第2-4行即可,这样jboss在启动时,会把这个当成全局module,默认加载。
注:这种方式虽然简单,但是不推荐,原因是如果jboss部署了多个应用,其它应用不知道有这个全局module,在自己的项目中再重复打包这些jar包,极容易在启动时造成冲突,最终启动失败,各种报错。
只有一种情况,比较适合这种方式:数据库驱动jar,比如:ojdbc6.jar这类,不过对于数据库驱动jar包而言,有一种更简单的方式,直接将驱动jar包当成普通应用来部署就行了,所以这种改standalone.xml的方式,仍然体现不出优越性。
1.5 修改pom.xml ,不打包公用jar包
既然mylib这个公用jar已经移动jboss中了,那么myweb这个项目打包里就不必再打包它了,修改pom.xml中的相关部分:
12 yjmyzz 3mylib 41.0 5provided 6
注意:第5行,改成provided表示该依赖项,运行时由jboss容器提供,因此打包时该项被忽略。
二、进阶篇
根据前面的步骤,好象并不难弄,但是实际应用中,如果想把一些知名的开源jar包,比如:spring, mybatis, hibernate , jackson这些全都以module的形式弄到jboss里,却并不容易,原因在于:这些开源项目本身又依赖其它开源项目,而其它开源项目,还有依赖,一层一层分析下来,关系十分复杂。更要命的的jboss本身也默认集成了一些知名的开源项目,比如:commons-logging之类,所以相互之间极容易冲突。就拿spring来说,最低层的是spring-core,它是依赖项最少的,如果要放到jboss的module中,module.xml内容为:
1 2 34 5 6 8 97 10 14 1511 12 13
注意:9-13行,这里表示spring-core依赖了哪些更底层次的module,一个也都不能少,否则启动时就会有一堆其名其妙的错误。所以,分析众多开源项目和jboss内置module的依赖关系,将是一个极大的挑战。
spring 4.1.1-RELEASE整套jar包,全部以module方式融合进jboss的话,步骤如下:
a) modules目录下先创建子目录org/springframework/spring/mainb) 将spring的一堆jar包复制进来,文件列表如下:
spring-aop-4.1.1.RELEASE.jar spring-aspects-4.1.1.RELEASE.jar spring-beans-4.1.1.RELEASE.jarspring-context-4.1.1.RELEASE.jar spring-context-support-4.1.1.RELEASE.jar spring-core-4.1.1.RELEASE.jar spring-expression-4.1.1.RELEASE.jar spring-instrument-4.1.1.RELEASE.jar spring-instrument-tomcat-4.1.1.RELEASE.jar spring-jdbc-4.1.1.RELEASE.jarspring-jms-4.1.1.RELEASE.jarspring-messaging-4.1.1.RELEASE.jarspring-orm-4.1.1.RELEASE.jarspring-oxm-4.1.1.RELEASE.jarspring-test-4.1.1.RELEASE.jarspring-tx-4.1.1.RELEASE.jarspring-web-4.1.1.RELEASE.jarspring-webmvc-4.1.1.RELEASE.jarspring-webmvc-portlet-4.1.1.RELEASE.jarspring-websocket-4.1.1.RELEASE.jarc) module.xml内容如下:
1 23 %4 255 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 26 3127 28 29 30
d) 应用中jboss-deployment-structure.xml的内容如下:
1 23 4 85 76
三、实战篇
下面就以一个Spring MVC 4.1.1 RELEASE + mybatis 3.2.8 + mysql + druid 的实例来说明,如何定制共享jar包:
这是打包后WEB-INF/lib下的所有依赖jar包:aopalliance-1.0.jar
commons-dbcp-1.4.jarcommons-logging-1.1.3.jarcommons-pool-1.5.4.jarmybatis-3.2.8.jarmybatis-spring-1.2.2.jarmysql-connector-java-5.1.25.jarspring-aop-4.1.1.RELEASE.jarspring-beans-4.1.1.RELEASE.jarspring-context-4.1.1.RELEASE.jarspring-core-4.1.1.RELEASE.jarspring-expression-4.1.1.RELEASE.jarspring-jdbc-4.1.1.RELEASE.jarspring-tx-4.1.1.RELEASE.jarspring-web-4.1.1.RELEASE.jarspring-webmvc-4.1.1.RELEASE.jar整个应用打包后,war包的尺寸约为6.7M(注:druid数据源是用在jboss创建jndi datasource的,所以不需要打包在war中)。根据前面的介绍,一步一步来精减发布包:1. %JBOSS_HOME/modules/mylib/main 先创建该目录,把lib下的这些jar文件,全复制进去
2. 然后创建module.xml
1 23 4 215 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 22 2923 24 25 26 27 28
3. 应用中的jboss-deployment-structure.xml
1 23 4 115 6 78 109
4. META-INF的特殊处理
因为spring的相关jar全放到jboss中了,这样会给应用本身运行带来问题,spring程序在启动时,会解析bean.xml配置文件,这个过程会加载spring.handers等文件,原来spring打包在应用本身中时,这些文件内嵌在sping的jar中,所以不会有问题,现在这些文件没有了,解析过程就会报错,因此需要手动把这些文件放到META-INF下,如图:
图中的这些文件,在spring里都可以找到,如果你的应用还使用了struts2,jstl,standard这些jar包,这些jar包里META-INF下的tld等文件也要复制到自己项目的META-INF下,根据我实际测试的结果,如果出现问题
spring-webmvc-xxx.jar
struts2-core-xxx.jar
这二个jar最好还是打包到应用中
最后别忘记了修改pom.xml文件,把所有依赖项的scope改成provided。
这样处理后,war包的尺寸从6.7M直接降到19K。