以下整理几个实际使用碰到的问题
移除spring core ,在插件项目中对于spring 的依赖设置为provide
与上边一个问题一样,不然会有莫名的问题
官方提供了SpringPluin 抽象 ,此类可以解决spring context 的创建以及一些依赖处理,一般的玩法是自己创建一个,同时对于需要
注意的beans在此对象中注册
参考
PluginBConfig 为一个Configuration ,可以声明我们需要的bean
public class PluginB extends SpringPlugin {
public PluginB(PluginWrapper wrapper) {
super(wrapper);
}
@Override
protected ApplicationContext createApplicationContext() {
AnnotationConfigApplicationContext applicationContext = new AnnotationConfigApplicationContext();
applicationContext.setClassLoader(getWrapper().getPluginClassLoader());
applicationContext.register(PluginBConfig.class);
applicationContext.refresh();
return applicationContext;
}
}
protected void registerExtension(Class<?> extensionClass) {
Map<String, ?> extensionBeanMap = springPluginManager.getApplicationContext().getBeansOfType(extensionClass);
if (extensionBeanMap.isEmpty()) {
Object extension = springPluginManager.getExtensionFactory().create(extensionClass);
beanFactory.registerSingleton(extensionClass.getName(), extension); // 使用了bootstrap 的applicationcontext
} else {
log.debug("Bean registeration aborted! Extension '{}' already existed as bean!", extensionClass.getName());
}
}
此扩展在bootstrap 的applicationcontext 中注册,对于插件同时也包含一个createApplicationContext,在此处我们可以i基于
spring 的context 父子关系解决插件共享bootstrap bean 的问题
参考代码
public class pluginC extends SpringPlugin {
public pluginC(PluginWrapper wrapper) {
super(wrapper);
}
@Override
protected ApplicationContext createApplicationContext() {
// 先获取SpringPluginManager 的ApplicationContext,也就是bootstrap 的ApplicationContext
ApplicationContext applicationContextRoot = ((SpringPluginManager)getWrapper().getPluginManager()).getApplicationContext();
AnnotationConfigApplicationContext applicationContext = new AnnotationConfigApplicationContext();
applicationContext.setClassLoader(getWrapper().getPluginClassLoader());
// 设置父applicationtext
applicationContext.setParent(applicationContextRoot);
applicationContext.register(PluginCConfig.class);
applicationContext.refresh();
return applicationContext;
}
}
注意一个问题,将spring 依赖设置为provide,不然会有上边2的问题,注意共享bean是子能访问父,父不能访问子(spring 约定)
如果需要访问,推荐放到服务契约中,在插件中开发实现,让SpringPluginManager 自己进行bean 注册到父中,而且此bean在插件中
也可以直接访问
https://github.com/pf4j/pf4j/issues/93
https://github.com/rongfengliang/pf4j-spring-learning/tree/classloader-test