解決HSF在JETTY中不能正常使用的问题

写了一个main函数

	public static void main(String[] arg) throws Exception {
		HSFContainerDelegator.start("D:\\j2ee_solution\\hsf-plugin");

		System.out.println(Thread.currentThread().getContextClassLoader());

		Class<?> clazz = Class
				.forName("com.taobao.hsf.app.spring.util.HSFSpringConsumerBean");

		System.out.println(clazz);
	}

 

不在当前classpath中是通过osgi容器加载进来的,这个在main函数中能够正常使用,但是当使用jetty作为容器的web服务启动的时候,通过Spring来加载,就会抛出异常,如下:

public class HsfBootup implements InitializingBean {

	@Override
	public void afterPropertiesSet() throws Exception {
		HSFContainerDelegator.start("D:\\j2ee_solution\\hsf-plugin");
		Class<?> clazz = Class
				.forName("com.taobao.hsf.app.spring.util.HSFSpringConsumerBean");

		System.out.println("======================" + clazz);
	}}

 

同样的代码,前一个是通过main函数启动,后一个是通过jetty间接使用spring来启动,但是后者会抛出异常。

为什么会这样呢?

那先看看两个场景下classloader有啥不同:

  1. main函数中的classloader是 sun.misc.Launcher$AppClassLoader@19821f
  2. jetty容器来加载的classloader是 class org.mortbay.jetty.webapp.WebAppClassLoader 

恩,看来两个场景下的classloader是不一样的,应该是classloader不一样导致的class加载不到。

再来看一下第二个场景下因为加载不到class而抛出的异常信息:

java.lang.ClassNotFoundException: com.taobao.hsf.app.spring.util.HSFSpringConsumerBean
        at java.net.URLClassLoader$1.run(URLClassLoader.java:202)
        at java.security.AccessController.doPrivileged(Native Method)
        at java.net.URLClassLoader.findClass(URLClassLoader.java:190)
        at java.lang.ClassLoader.loadClass(ClassLoader.java:307)
        at org.codehaus.classworlds.RealmClassLoader.loadClassDirect(RealmClassLoader.java:195)
        at org.codehaus.classworlds.DefaultClassRealm.loadClass(DefaultClassRealm.java:255)
        at org.codehaus.classworlds.DefaultClassRealm.loadClass(DefaultClassRealm.java:274)
        at org.codehaus.classworlds.RealmClassLoader.loadClass(RealmClassLoader.java:214)
        at java.lang.ClassLoader.loadClass(ClassLoader.java:248)
        at org.mortbay.jetty.webapp.WebAppClassLoader.loadClass(WebAppClassLoader.java:401)
        at org.mortbay.jetty.webapp.WebAppClassLoader.loadClass(WebAppClassLoader.java:363)
        at org.springframework.util.ClassUtils.forName(ClassUtils.java:211)
        at org.springframework.beans.factory.support.AbstractBeanDefinition.resolveBeanClass(AbstractBeanDefinition.java:385)
        at org.springframework.beans.factory.support.AbstractBeanFactory.resolveBeanClass(AbstractBeanFactory.java:1138)
        at org.springframework.beans.factory.support.AbstractAutowireCapableBeanFactory.predictBeanType(AbstractAutowireCapableBeanFactory.j
ava:524)
        at org.springframework.beans.factory.support.AbstractBeanFactory.isFactoryBean(AbstractBeanFactory.java:1177)
        at org.springframework.beans.factory.support.AbstractBeanFactory.isFactoryBean(AbstractBeanFactory.java:758)
        at org.springframework.beans.factory.support.DefaultListableBeanFactory.preInstantiateSingletons(DefaultListableBeanFactory.java:422
)
        at org.springframework.context.support.AbstractApplicationContext.finishBeanFactoryInitialization(AbstractApplicationContext.java:72
8)
        at org.springframework.context.support.AbstractApplicationContext.refresh(AbstractApplicationContext.java:380)
        at org.springframework.web.context.ContextLoader.createWebApplicationContext(ContextLoader.java:255)
        at org.springframework.web.context.ContextLoader.initWebApplicationContext(ContextLoader.java:199)
        at org.springframework.web.context.ContextLoaderListener.contextInitialized(ContextLoaderListener.java:45)
        at org.mortbay.jetty.handler.ContextHandler.startContext(ContextHandler.java:548)
        at org.mortbay.jetty.servlet.Context.startContext(Context.java:136)

 在网上找到了jetty的classloader加载机制http://docs.codehaus.org/display/JETTY/Classloading ,这其中说到了org.mortbay.jetty.webapp.WebAppClassLoader是jetty的默认的classloader。

你可能感兴趣的:(jetty)