web应用中加入osgi支持

osgi是个好东东,网上有一大堆说词。



osgi有主要有两大好处:1,模块化 2,动态化

如果想模块化,那个整个系统架构都要构建于osgi基础之上,这不但要求架构师对osgi有相当的认识,对模块化的合理抽象能力,而且要求开发成员对osgi有一定的认识。现在还没有那个条件。

2、动态化,在7*24系统中,如果需要更改业务处理逻辑,怎么办?重新启动中间件,肯定不行。当然可以用其它方法,如水平集群。但是显得不那么自然。而osgi本身就具有动态加载的能力,说白了就是java的类加载器进行。

那怎样做到动态加载,更新,删除服务?
我们可以在web中加入osgi支持,两种方法:web服务器嵌入到osgi内核中、利用bridge将osgi嵌入到web中。我的方法是在web应用中启动osgi内核。

在应用的中加入启动osgi的listener.

<listener>
  <description>start osgi</description>
  <listener-class>
   com.dx.frame.web.listener.OsgiListener
  </listener-class>
</listener>

package com.dx.frame.web.listener;

import javax.servlet.ServletContextEvent;
import javax.servlet.ServletContextListener;

import org.eclipse.core.runtime.adaptor.EclipseStarter;
import org.osgi.framework.BundleContext;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import com.dx.frame.bundles.OsgiBundlesContext;

/**
* 在应用中间件中启动OSGI
*
* @author zhxp
*
*/
public class OsgiListener implements ServletContextListener {

private Logger logger = LoggerFactory.getLogger(getClass());

public void contextDestroyed(ServletContextEvent event) {
  try {
   EclipseStarter.shutdown();
   OsgiBundlesContext.clear();
   logger.info("osgi 停止完成");
  } catch (Exception e) {
   // TODO Auto-generated catch block
   e.printStackTrace();
  }
}

@Override
public void contextInitialized(ServletContextEvent arg0) {
  // TODO Auto-generated method stub

  try {
  
   ClassLoader loader = OsgiListener.class.getClassLoader();
   EclipseStarter.startup(new String[] { "-console" }, null);
   BundleContext context = EclipseStarter.getSystemBundleContext();
   OsgiBundlesContext.setOsgiContxt(context);
   logger.info("osgi 启动完成");
   //注意设置ClassLoader,不然在非osgi后续类加载中(有些会用当前线程的类加载器,而此时类加载器已经被OSGI修改???)会使用到osgi的类加载器
   Thread.currentThread().setContextClassLoader(loader);
  } catch (Exception e) {
   // TODO Auto-generated catch block
   e.printStackTrace();
  }

}

}
这时可以通过控制台进行osgi命令操作。

在我们的应用中如果需要osgi服务中能通过反射机制才能正常调用。

你可能感兴趣的:(eclipse,spring,应用服务器,Web,osgi)