此想法构思于去年某个时候,并于去年的某个时候后的某个时候动手实践,终于于今年有点眉目了,以此开题来纪念我的懒惰-_-!。
书归正传,eclipse已为大家所熟知,其框插件化的框架为大家所称赞。但这一切都是在应用程序中实现的,假如WEB程序中也能如此的话,那将是一个多么美好的春天啊,试想不用再为了某一个小模块而更新整个程序(尤其是某些应用预编译技术而又没做明确模块划分的WEB应用),不再为了删除客户用不到的模块而去代码的每一处仔细的查找关联(我承认以上问题都是我自己犯的 -_-!)。假如手里有足够的代码模块,一个WEB程序也许须要修改的地方仅有前台界面而已(只相对于网站程序而言,因为比较简单)。
声明:这个只是为了能运行而改,源码修改部分只是最简单的修改,当然也有水平问题,代码不会很优雅。对于修改后框架的没有其它影响不得而知,这只是我一个想法的实践而已,能不能用到实际处还需要做出一个东西检验一下,
而且spring已经有一个基于OSGI技术的框架,和eclipse框架基本同一种技术的web插件化框架SpringSource dm Server。
1、准备
tomcat eclipse spring源码 jdk1.5+
2、思想
俗话说的好,没有做不到,只有想不到,先了解下此种实现的想法,这也是为了自己的备忘。
实现过程:
1、确定插件目录结构;
2、扫描插件目录;
3、加载非lib下的JAR文件;
4、加载struts、spring及mapping配置文件;
很简单的思想,对吧。
3、实现思想
1、确定插件目录结构;
参照eclipse的设计,应当有一个总的配置文件,为了简单,这份配置文件上主要体现的是各需要用到的资源的路径。
除了一个配置清单外,要有一个总的插件目录,来存放各个插件,每一个插件还要有一个自己的目录,对于插件自己的目录而言,
里的东西可以随便放了,因为要用的都要写的配置清单上的。
以下为我的插件配置文件:
2、扫描目录
这个非常简单的IO操作我就不多说了,主要完成的工作是找到每一个插件的配置清单。
3、tomcat启动加载插件目录下的jar文件
对于应该程序而言,你们可以自己写个classloader,然后用这个classloader加载jar,但对于web应用而言,这种方式理论是
不可行的,因为运行环境已变更,应用程序是运行在JAVA虚拟机上,而WEB程序是运行是WEB容器上。
查找相关资料后知道,TOMCAT加载JAR文件分3个CLASSLOADER,我们要用到的是最后一个CLASSLOADER,为WebappClassLoader。对于调用WebappClassLoader只需要两句话就行了
要让TOMCAT启动运行我们的加载JAR程序,可以让程序实现ServletContextListener接口,当然还有别的接口可以实现相同的功能,最后在web.xml将我们的程序注册为一个LISTENER。
4、加载struts、spring及mapping配置文件
通过扫描目录,我们已将插件的配置清单找到,对于解析我就不多说了,对于XML文件,JDOM和DOM4J都是好东西。
加载struts
在正常的配置情况下,在web.xml就为struts配置一个SERVLET,为org.apache.struts.action.ActionServlet,我们要做的就是将继承它,并覆盖init()方法。
先给出org.apache.struts.action.ActionServlet中init()的源代码:
我的做法是在这个方法里再加一个循环,可能很笨,但好用啊
代码里的strutsString为读出的插件中的struts配置文件路径,数据结构为 path,path 的字符串。
加载spring配置文件
此处让我费了不少事,水平太洼,没办法的事 -_-!。
对于spring需要修改3处,serveice bean的加载,mapping的加载。
由于spring里全是接口,单单继承是没法解决问题的,所以只能修改源代码了。修改代哪部分代码可以达到我们的目的,web.xml和applicationContext.xml文件中找到很好的答案。
启动文件:org.springframework.web.context.ContextLoaderListener、加载mapping的文件:org.springframework.orm.hibernate3.LocalSessionFactoryBean。由ContexLoaderListener可追查到加载bean的文件为ContextLoader。下面为修改后的代码:
其中最关键的为倒数第3行的代码,参数configString即为表示spring配置文件路径的参数,是结构为 path,path 的字符串。try...catch中为我自己加的代码,
首先会判断解析插件配置清单的类是否存在,不如不存在则不跳过插件中的bean配置,如果存在解析类的话,则加载插件中的bean配置文件。
加载mapping
打开LocalSessionFactoryBean你会发现在有如下变量:
很眼熟是吧,找到下面一堆if...else,你用的哪个配置就改对应的源码吧,因为我用的mappingJarLocations,所以我的修改如下:
Resource数组中存的都是ServletContextResource类型的对象,千万别弄错了哦。
最后是加载plug-in,因为struts也支持模块化配置,可以将一份配置文件分成好几分,但其实中的plug-in配置并不是共享的,所以为每一份sturts配置文件都写上plug-in,既然每一份配置文件都是独立,那自己也就不用修改源码了。
至此,struts+spring+hibernate的插件化也修改完毕了,文采不佳,有幸看到此文章的朋友们将就着看吧。
PS:spring的源码不看不知道,一看吓一跳啊,我突然觉得我就是一个会拿电脑打打字,玩玩游戏,听听歌,看看电影的人。JAVA是什么,编程是什么,能吃么,好吃么 -_-!