一个简单的osgi实现框架,可以动态装载运行,jar包。
public static void main(String[] args) throws Exception {
//模块名称
final String module="comparedb";
//获取OSGI容器实例
final OSGiContainer container = OSGiContainer.getInstance();
//启动框架
container.start();
//设置模块安装成功后的处理
container.setModuleInstalledHandler(new ModuleInstalledHandler() {
@Override
public void moduleInstalled(ModuleLoader loader) {
//模块安装成功后运行该模块
Thread.currentThread().setContextClassLoader((ClassLoader)container.getLoader(module));
container.run(module);
}
});
//创建模块模型
ModulesBean modulesBean = new ModulesBean(module, "1.0", new JarFile[]{new JarFile("/codes/myproject/CompareDB/target/CompareDB.jar")},new MainClassRunner("com.hyberbin.main.Main"));
//从本地maven仓库中自动解决依赖
modulesBean.setModuleLoader(new MavenModuleLoaderImpl("/Users/hyberbin/.m2/repository/"));
//安装
container.install(modulesBean);
}
spring-dynamic文件夹下是动态安装一个SpringMVC模块的实现
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns:context="http://www.springframework.org/schema/context"
xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-3.2.xsd
http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-3.2.xsd"
default-lazy-init="true">
<bean class="org.springframework.orm.hibernate4.HibernateExceptionTranslator"/>
<!--<bean class="org.springframework.dao.annotation.PersistenceExceptionTranslationPostProcessor"/>-->
<!--自己实现了一个PersistenceExceptionTranslationPostProcessor,原来的获取ClassLoader的方式有问题-->
<bean class="com.XXX.spring.dynamic.aop.UserPersistenceExceptionTranslationPostProcessor"/>
<!--让这个sessionFactory来动态路由多个sessionFactory-->
<bean id="sessionFactory" class="com.XXX.spring.dynamic.factory.DynamicSessionFactory"/>
<!--这个是将所有的sessionFactory放入Map中-->
<bean id="sessionFactoryMap" class="java.util.HashMap">
<constructor-arg>
<map>
<entry key="defaultSessionFactory">
<ref bean="defaultSessionFactory"/>
</entry>
</map>
</constructor-arg>
</bean>
<!-- Hibernate SessionFactory -->
<bean id="defaultSessionFactory" lazy-init="true" class="org.springframework.orm.hibernate4.LocalSessionFactoryBean" destroy-method="destroy">
<property name="dataSource" ref="dataSource"/>
<property name="configLocation" value="/WEB-INF/hibernate.cfg.xml"/>
<property name="packagesToScan">
<list>
<value>com.XXX.digitalcampus.**.model</value>
</list>
</property>
<property name="hibernateProperties">
<props>
<prop key="hibernate.dialect">${hibernate.dialect}</prop>
<prop key="hibernate.show_sql">false</prop>
<prop key="hibernate.format_sql">false</prop>
<prop key="hibernate.query.substitutions">true 'Y', false 'N'</prop>
<prop key="hibernate.cache.use_second_level_cache">true</prop>
<prop key="hibernate.cache.provider_class">org.hibernate.cache.ehcache.EhCacheRegionFactory</prop>
<prop key="javax.persistence.validation.mode">none</prop>
<!--<prop key="hibernate.current_session_context_class">org.springframework.orm.hibernate4.SpringSessionContext</prop>-->
<!--<prop key="hibernate.transaction.auto_close_session">true</prop>-->
<!-- Hibernate Search index directory -->
<!--<prop key="hibernate.search.default.indexBase">${app.search.index.basedir}</prop>-->
</props>
<!-- Turn batching off for better error messages under PostgreSQL -->
<!-- hibernate.jdbc.batch_size=0 -->
</property>
</bean>
<!-- Transaction manager for a single Hibernate SessionFactory (alternative to JTA) -->
<!--<bean id="txManager" class="org.springframework.orm.hibernate4.HibernateTransactionManager">-->
<!--<property name="sessionFactory" ref="sessionFactory"/>-->
<!--</bean>-->
<!--动态事务管理器-->
<bean id="txManager" class="com.XXX.spring.dynamic.factory.DynamicTransactionManager">
<property name="sessionFactory" ref="sessionFactory"/>
</bean>
<!-- Activates scanning of @Repository -->
<context:component-scan base-package="com.XXX.framework.**.data.hibernate"/>
<context:component-scan base-package="com.XXX.digitalcampus.**.dao.hibernate"/>
</beans>
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns:aop="http://www.springframework.org/schema/aop"
xmlns:context="http://www.springframework.org/schema/context"
xmlns:tx="http://www.springframework.org/schema/tx"
xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-3.2.xsd
http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop-3.2.xsd
http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-3.2.xsd
http://www.springframework.org/schema/tx http://www.springframework.org/schema/tx/spring-tx-3.2.xsd"
default-lazy-init="true">
<!-- =================================================================== -->
<!-- AOP: Configuration and Aspects -->
<!-- =================================================================== -->
<tx:advice id="txAdvice" transaction-manager="txManager">
<tx:attributes>
<tx:method name="save*" rollback-for="com.XXX.framework.core.exception.BusinessException"/>
<tx:method name="add*" rollback-for="com.XXX.framework.core.exception.BusinessException"/>
<tx:method name="create*" rollback-for="com.XXX.framework.core.exception.BusinessException"/>
<tx:method name="insert*" rollback-for="com.XXX.framework.core.exception.BusinessException"/>
<tx:method name="update*" rollback-for="com.XXX.framework.core.exception.BusinessException"/>
<tx:method name="merge*" rollback-for="com.XXX.framework.core.exception.BusinessException"/>
<tx:method name="del*" rollback-for="com.XXX.framework.core.exception.BusinessException"/>
<tx:method name="remove*" rollback-for="com.XXX.framework.core.exception.BusinessException"/>
<tx:method name="put*" rollback-for="com.XXX.framework.core.exception.BusinessException"/>
<tx:method name="execute*" rollback-for="com.XXX.framework.core.exception.BusinessException"
no-rollback-for="java.lang.RuntimeException"/>
<tx:method name="clear*" rollback-for="com.XXX.framework.core.exception.BusinessException"
no-rollback-for="java.lang.RuntimeException"/>
<tx:method name="disable*" rollback-for="com.XXX.framework.core.exception.BusinessException"
no-rollback-for="java.lang.RuntimeException"/>
<tx:method name="use*" rollback-for="com.XXX.framework.core.exception.BusinessException"
no-rollback-for="java.lang.RuntimeException"/>
<!--hibernate4必须配置为开启事务 否则 getCurrentSession()获取不到-->
<tx:method name="get*" read-only="true"/>
<tx:method name="count*" read-only="true"/>
<tx:method name="find*" read-only="true"/>
<tx:method name="list*" read-only="true"/>
<tx:method name="*" read-only="false"/>
</tx:attributes>
</tx:advice>
<!--这个modulesAdvice主要是为了在ThreadLocal中放入一个模块名,后面好获取对应模块的sessionFactory-->
<bean id="modulesAdvice" class="com.XXX.spring.dynamic.aop.ServiceMethodBeforeAdvice"/>
<!--<!– 配置pointcut,将事务通过aop方式插入到joinpoint –>-->
<!--<aop:config>-->
<!--<aop:pointcut expression="execution(* com.XXX..service.impl.*.*(..))" id="serviceMethod"/>-->
<!--<!–这个modulesAdvice主要是为了在ThreadLocal中放入一个模块名,后面好获取对应模块的sessionFactory–>-->
<!--<aop:advisor advice-ref="modulesAdvice" pointcut-ref="serviceMethod"/>-->
<!--<aop:advisor advice-ref="txAdvice" pointcut-ref="serviceMethod"/>-->
<!--</aop:config>-->
<!--<aop:aspectj-autoproxy proxy-target-class="true"/>-->
<!--这里没用<aop:>这种高级的配置因为AutoProxyCreator是我自己实现的,之前的AutoProxyCreator不能获取当前的ClassLoader-->
<bean id="autoProxyCreator" class="com.XXX.spring.dynamic.aop.AutoProxyCreator"></bean>
<!--第一个注入modulesAdvice-->
<bean id="aspectjAspect" class="org.springframework.aop.aspectj.AspectJExpressionPointcutAdvisor">
<property name="advice" ref="modulesAdvice"/>
<property name="expression" value="execution(* com.XXX..service.impl.*.*(..))"/>
</bean>
<!--第二个注入modulesAdvice,注入事务代理,要从第一个切入点中获取当前模块的SessionFactory-->
<bean id="aspectjAspect2" class="org.springframework.aop.aspectj.AspectJExpressionPointcutAdvisor">
<property name="advice" ref="txAdvice"/>
<property name="expression" value="execution(* com.XXX..service.impl.*.*(..))"/>
</bean>
<!--osgi模块的卸载器-->
<bean id="moduleUnInstaller" class="com.XXX.spring.dynamic.uninstaller.SpringModuleUninstall"/>
<!--启动OSGI容器-->
<bean id="osgi-container" class="org.jplus.osgi.core.OSGiContainer" factory-method="getInstance" lazy-init="false" init-method="start">
<property name="moduleUnInstaller" ref="moduleUnInstaller"/>
</bean>
<!-- Activates scanning of @Service -->
<context:component-scan base-package="com.XXX.framework.**.service"/>
<context:component-scan base-package="com.XXX.digitalcampus.**.service"/>
<!-- Configure Velocity for sending e-mail -->
<bean id="velocityEngine" class="org.springframework.ui.velocity.VelocityEngineFactoryBean">
<property name="velocityProperties">
<props>
<prop key="resource.loader">class</prop>
<prop key="class.resource.loader.class">
org.apache.velocity.runtime.resource.loader.ClasspathResourceLoader
</prop>
<prop key="velocimacro.library"></prop>
</props>
</property>
</bean>
</beans>
作者已经实现了用这个框架动态加载自己项目中的模块.
- 动态装载Controller类,并注册BeanFactory缓存服务.
- 动态装载Service类,并注册Aop服务和BeanFactory缓存服务,开启Hibernate事务.
- 动态装载Hibernate类,并动态生成和加载新的SessionFactory,和BeanFactory缓存服务.
- 自动将jar包中的jsp文件解压到web容器下.
- 可以自动结束该模块创建的所有线程.
- 可以自动结束模块中带有Frame窗体的程序.