Spring Web Application应用到OSGI环境 三

一、Spring Framework
通常Spring的web应用结构层次:
1,ContextLoaderServlet,创建一个ContextLoader做为整个web应用的Loader,它会为自己创建一个web应用级别的WebApplicationContext,接着默认加载applicationContext.xml里面的bean
2,DispatcherServlet,同样创建一个WebApplicationContext,加载dispatch-bean,它会将ContextLoader的WebApplicationContext做为自己的父ApplicationContext

二、Spring-OSGI
通过Spring-osgi规范可以了解到spring-osgi会为每一个有spring-bean配置文件的bundle创建一个ApplicationContext,spring-bean配置文件里面的bean则被它加载,所以Dispatch-bean-loader无法找到spring-osgi-bean

三、RESOLVER
首先要先得到bundle的ApplicationContext,然后创建一个自定义的ContextLoader-BundleContextLoader,不加载配置文件,重写loadParentContext方法,将父ApplicationContext设置成bundle的ApplicationContext,然后生成一个自定义的WebApplicationContext-BundleWebApplicationContext,里面的bean信息直接从parentApplicationContext里面获取。本来spring规范里面写着org.springframework.osgi.context.support.WebApplicationContext ,但是找个底朝天,也没找到,可能还没出呢吧,于是只能自己写了。
bundle的ApplicationContext可通过org.springframework.context.ApplicationContextAware.setApplicationContext(ApplicationContext applicationContext)得到

四、BundleContextLoaderServlet
在org.phrancol.osgi.jpetstore.springmvc里新建类BundleContextLoaderServlet
public   class  BundleContextLoaderServlet  extends  ContextLoaderServlet  {
    
private final ApplicationContext applicationContext;
    
private BundleContext ctx;

    
public BundleContextLoaderServlet(BundleContext ctx,
            ApplicationContext appContext) 
{
        
this.ctx = ctx;
        
this.applicationContext = appContext;
    }

    
    
/** *//**
     * 重写该方法,用于构造一个自定义的BundleContextLoader
     
*/

    
protected ContextLoader createContextLoader() {
        
return new BundleContextLoader(applicationContext, ctx);
    }

}
五、BundleContextLoader
在org.phrancol.osgi.jpetstore.springmvc里新建类BundleContextLoader
public   class  BundleContextLoader  extends  ContextLoader  {
    
private final ApplicationContext applicationContext;
    
private BundleContext bundleContext;

    
public BundleContextLoader(ApplicationContext appContext, BundleContext ctx) {
        
this.applicationContext = appContext;
        
this.bundleContext = ctx;
    }

    
    
/** *//**
     * 重写该方法,让其返回的ParentContext是Bundle里的那个ApplicationContext
     
*/

    
protected ApplicationContext loadParentContext(ServletContext servletContext)
            
throws BeansException {
        
return applicationContext;
    }


    
/** *//**
     * 生成一个自定义的BundleWebApplicationContext,返回的bean信息直接从Parent获取
     * 这里的代码请参考org.springframework.osgi.extender.support.ApplicationContextCreator.run()方法里的代码
     
*/

    
protected WebApplicationContext createWebApplicationContext(
            ServletContext servletContext, ApplicationContext parent)
            
throws BeansException {
        ClassLoader contextClassLoader 
= Thread.currentThread()
                .getContextClassLoader();
        
try {
            ClassLoader cl 
= BundleDelegatingClassLoader
                    .createBundleClassLoaderFor(bundleContext.getBundle(), getClass()
                            .getClassLoader());
            Thread.currentThread().setContextClassLoader(cl);
            LocalBundleContext.setContext(bundleContext);
            
return new BundleWebApplicationContext(parent, servletContext);
        }
 finally {
            Thread.currentThread().setContextClassLoader(contextClassLoader);
        }

    }

}

六、BundleWebApplicationContext
在org.phrancol.osgi.jpetstore.springmvc里面新建类BundleWebApplicationContext,所有的方法都返回parent的方法返回值
public   class  BundleWebApplicationContext  implements  WebApplicationContext  {
    
    
private final ApplicationContext parentApplicationContext;
    
private ServletContext servletContext;
    
    
public BundleWebApplicationContext(ApplicationContext parentApplicationContext,
            ServletContext servletContext)
{
        
this.parentApplicationContext = parentApplicationContext;
        
this.servletContext = servletContext;
    }


    
public ServletContext getServletContext() {
        
// TODO Auto-generated method stub
        return servletContext;
    }


    
public AutowireCapableBeanFactory getAutowireCapableBeanFactory()
            
throws IllegalStateException {
        
// TODO Auto-generated method stub
        return parentApplicationContext.getAutowireCapableBeanFactory();
    }


    
public String getDisplayName() {
        
// TODO Auto-generated method stub
        return parentApplicationContext.getDisplayName();
    }

        .
}

七、修改
1,修改接口HttpServiceRegister.serviceRegister(BundleContext context),改成
serviceRegister(BundleContext context, ApplicationContext bundleApplicationContext);
2,修改SpringmvcHttpServiceRegister.serviceRegister(BundleContext context),改成
serviceRegister(BundleContext context, ApplicationContext bundleApplicationContext)
3,SpringmvcHttpServiceRegister.serviceRegister方法修改一下
public   void  serviceRegister(BundleContext context, ApplicationContext bundleApplicationContext)  {
        
try {
            ServiceReference sr 
= context.getServiceReference(HttpService.class
                    .getName());
            HttpService httpService 
= (HttpService) context.getService(sr);
            httpService.registerResources(
"/""/web"null);
            httpService.registerServlet(
"/*.jsp"new JspServlet(context
                    .getBundle(), 
"/web"), nullnull);
            Dictionary
<String, String> initparams = new Hashtable<String, String>();
            initparams.put(
"load-on-startup""1");
           
/*  编辑器的问题,这4行代码不是注释。。。 */
            ContextLoaderServlet contextloaderListener = new BundleContextLoaderServlet(
                    context, bundleApplicationContext);
            httpService.registerServlet("/ContextLoader",
                    contextloaderListener, initparams, null);
            /*   
*/

            DispatcherServlet dispatcherServlet 
= new DispatcherServlet();
            dispatcherServlet
                    .setContextConfigLocation(
"META-INF/dispatcher/petstore-servlet.xml");
            initparams 
= new Hashtable<String, String>();
            initparams.put(
"servlet-name""petstore");
            initparams.put(
"load-on-startup""2");
            httpService.registerServlet(
"/*.do", dispatcherServlet, initparams,
                    
null);
        }
 catch (Exception e) {
            e.printStackTrace(System.out);
        }

    }

OK,这下应该没问题了,启动看看效果
倒,还是报错。。。。。。

八、HttpContext
来看看org.osgi.service.http.HttpService.registerServlet的代码
  * Servlets registered with the same <code>HttpContext</code> object will
  * share the same <code>ServletContext</code>.
原来是这样,注册了2个servlet,于是生成了2个不同的HttpContext,那就只生成一个试试
HttpContext defaultContext  =   new  DefaultHttpContext(context.getBundle());

httpService.registerServlet(
" /init-context-loader-petsore-web " ,
                    contextloaderListener, initparams, defaultContext);

httpService.registerServlet(
" /*.do " , dispatcherServlet, initparams,
                    defaultContext);

再启动,一切顺利,访问页面也正常。

九、打完收工
将org.phrancol.osgi.jpetstore.springmvc/METS-INF/dispatcher/petstore-servlet.xml里面的bean补全(注意还有2个bean需要注册和引用),再运行,就OK了。

 

 

 

 

你可能感兴趣的:(spring,Web,bean,servlet,osgi)