最近做了个有趣的事,今晚不想写代码,来记录一下,如果需要的人看见也算是功德一件。

可以先拉下来看看效果。

需求描述:产品被越来越多的用户使用,需求的差异也愈来愈大,为了适应不同用户的不同需求,将产品进行插件化升级。

目标:将项目拆分为主程序和插件,主程序感知不到插件的存在,插件作为独立的jar包放在某个位置。主程序启动时把所有需要的插件动态加载进来。开发插件与写主程序无异,主程序代码可以随时迁移到插件。

第一步:调研

最蛋疼的当然是依赖注入的问题啦。首要目标就是先解决插件中的类使用的主程序中由spring管理起来的类,以及插件中需要由spring管理起来的类的问题。

所以呢,先要看看spring是在什么时机进行bean的实例化的。

JAVA插件框架封装 java插件式 开发模块_初始化

入口程序瞧一瞧,逻辑清晰,步骤写得清清楚楚(虽然不知道有什么用)。好吧,每一步的源码再瞧一瞧,

JAVA插件框架封装 java插件式 开发模块_JAVA插件框架封装_02

嘻嘻,看到了listener,那是不是这里留有扩展呢?(别问我为啥)网上查资料,发现自己可以注册listener,在spring初始化完成之前,写入自己的逻辑,美滋滋。

把类交托给spring管理应该没啥问题了

接下来看看如何开发插件

要针对主程序进行开发,就要看看怎么能引用到主程序的资源了,我用的maven哈,打包插件

JAVA插件框架封装 java插件式 开发模块_java 插件式 开发_03

打出来的主程序,插件是引用不到主程序的类的。需要使用下面这个插件

JAVA插件框架封装 java插件式 开发模块_JAVA插件框架封装_04

插件开发应该没啥问题了

最后的问题就是主程序加载插件的问题

插件的类需要加载进主程序的classpath,可能部分类需要托管给spring管理,还有可能插件在spring初始化完成之后,自己进行初始化。

哇,好jr多的事。

思前想后,准备建一张插件表,其中配置插件的位置,插件的boot class,需要由spring管理的类的扫描路径。

主程序在启动时,读取插件表,加载打成jar包的插件,在spring初始化容器之前,根据扫描路径将需要由spring管理的类注册进BeanFactory,在spring初始化完成后,调用所有插件的boot class。