在Tomcat中部署JSF应用

众所周知,JSF1.2应用的官方Java应用服务器是Glassfish,网上很少有关于在Tomcat下部署JSF应用的例子。不信这个邪,我硬着头皮尝试在Tomcat中部署一个JSF应用。

 

我机器上Tomcat的版本是5.5的,将一个在Glassfish下测试通过的JSF应用放在webapp目录下,启动Tomact,错误马上出来:

SEVERE: Error configuring application listener of class com.sun.faces.config.GlassFishConfigureListener
java.lang.NoClassDefFoundError: javax/el/ExpressionFactory
 at java.lang.Class.getDeclaredConstructors0(Native Method)
 at java.lang.Class.privateGetDeclaredConstructors(Class.java:2328)
 at java.lang.Class.getConstructor0(Class.java:2640)
 at java.lang.Class.newInstance0(Class.java:321)
 at java.lang.Class.newInstance(Class.java:303)
 at org.apache.catalina.core.StandardContext.listenerStart(StandardContext.java:3713)
 at org.apache.catalina.core.StandardContext.start(StandardContext.java:4216)
 at org.apache.catalina.core.ContainerBase.addChildInternal(ContainerBase.java:760)
 at org.apache.catalina.core.ContainerBase.addChild(ContainerBase.java:740)
 at org.apache.catalina.core.StandardHost.addChild(StandardHost.java:544)
 at org.apache.catalina.startup.HostConfig.deployDirectory(HostConfig.java:920)
 at org.apache.catalina.startup.HostConfig.deployDirectories(HostConfig.java:883)
 at org.apache.catalina.startup.HostConfig.deployApps(HostConfig.java:492)
 at org.apache.catalina.startup.HostConfig.start(HostConfig.java:1138)
 at org.apache.catalina.startup.HostConfig.lifecycleEvent(HostConfig.java:311)
 at org.apache.catalina.util.LifecycleSupport.fireLifecycleEvent(LifecycleSupport.java:120)
 at org.apache.catalina.core.ContainerBase.start(ContainerBase.java:1022)
 at org.apache.catalina.core.StandardHost.start(StandardHost.java:736)
 at org.apache.catalina.core.ContainerBase.start(ContainerBase.java:1014)
 at org.apache.catalina.core.StandardEngine.start(StandardEngine.java:443)
 at org.apache.catalina.core.StandardService.start(StandardService.java:448)
 at org.apache.catalina.core.StandardServer.start(StandardServer.java:700)
 at org.apache.catalina.startup.Catalina.start(Catalina.java:552)
 at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
 at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:39)
 at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:25)
 at java.lang.reflect.Method.invoke(Method.java:585)
 at org.apache.catalina.startup.Bootstrap.start(Bootstrap.java:295)
 at org.apache.catalina.startup.Bootstrap.main(Bootstrap.java:433)

 

很简单的解决,我们的web应用缺少el-ri.jar和el-api.jar两个和el相关的jar包。之后,似乎一切都正常了,原来Tomcat部署JSF不过如此!可是当运行了我的http://localhost:8080/之后,一下子被弄懵了,一个既陌生又熟悉的错误被抛出来:

java.lang.NoClassDefFoundError: javax/servlet/jsp/tagext/JspIdConsumer
 java.lang.ClassLoader.defineClass1(Native Method)
 java.lang.ClassLoader.defineClass(ClassLoader.java:620)
 java.security.SecureClassLoader.defineClass(SecureClassLoader.java:124)
 org.apache.catalina.loader.WebappClassLoader.findClassInternal(WebappClassLoader.java:1847)
 org.apache.catalina.loader.WebappClassLoader.findClass(WebappClassLoader.java:873)
 org.apache.catalina.loader.WebappClassLoader.loadClass(WebappClassLoader.java:1326)
 org.apache.catalina.loader.WebappClassLoader.loadClass(WebappClassLoader.java:1205)
 java.lang.ClassLoader.loadClassInternal(ClassLoader.java:319)
 java.lang.ClassLoader.defineClass1(Native Method)
 java.lang.ClassLoader.defineClass(ClassLoader.java:620)
 java.security.SecureClassLoader.defineClass(SecureClassLoader.java:124)
 org.apache.catalina.loader.WebappClassLoader.findClassInternal(WebappClassLoader.java:1847)
 org.apache.catalina.loader.WebappClassLoader.findClass(WebappClassLoader.java:873)
 org.apache.catalina.loader.WebappClassLoader.loadClass(WebappClassLoader.java:1326)
 org.apache.catalina.loader.WebappClassLoader.loadClass(WebappClassLoader.java:1205)
 java.lang.ClassLoader.loadClassInternal(ClassLoader.java:319)
 java.lang.ClassLoader.defineClass1(Native Method)
 java.lang.ClassLoader.defineClass(ClassLoader.java:620)
 java.security.SecureClassLoader.defineClass(SecureClassLoader.java:124)
 org.apache.catalina.loader.WebappClassLoader.findClassInternal(WebappClassLoader.java:1847)
 org.apache.catalina.loader.WebappClassLoader.findClass(WebappClassLoader.java:873)
 org.apache.catalina.loader.WebappClassLoader.loadClass(WebappClassLoader.java:1326)
 org.apache.catalina.loader.WebappClassLoader.loadClass(WebappClassLoader.java:1205)
 org.apache.jasper.compiler.Parser.parseCustomTag(Parser.java:1326)
 org.apache.jasper.compiler.Parser.parseElements(Parser.java:1578)
 org.apache.jasper.compiler.Parser.parse(Parser.java:127)
 org.apache.jasper.compiler.ParserController.doParse(ParserController.java:212)
 org.apache.jasper.compiler.ParserController.parse(ParserController.java:101)
 org.apache.jasper.compiler.Compiler.generateJava(Compiler.java:156)
 org.apache.jasper.compiler.Compiler.compile(Compiler.java:296)
 org.apache.jasper.compiler.Compiler.compile(Compiler.java:277)
 org.apache.jasper.compiler.Compiler.compile(Compiler.java:265)
 org.apache.jasper.JspCompilationContext.compile(JspCompilationContext.java:564)
 org.apache.jasper.servlet.JspServletWrapper.service(JspServletWrapper.java:299)
 org.apache.jasper.servlet.JspServlet.serviceJspFile(JspServlet.java:315)
 org.apache.jasper.servlet.JspServlet.service(JspServlet.java:265)
 javax.servlet.http.HttpServlet.service(HttpServlet.java:803)
 com.sun.faces.context.ExternalContextImpl.dispatch(ExternalContextImpl.java:414)
 com.sun.faces.application.ViewHandlerImpl.executePageToBuildView(ViewHandlerImpl.java:455)
 com.sun.faces.application.ViewHandlerImpl.renderView(ViewHandlerImpl.java:139)
 com.sun.faces.lifecycle.RenderResponsePhase.execute(RenderResponsePhase.java:108)
 com.sun.faces.lifecycle.LifecycleImpl.phase(LifecycleImpl.java:266)
 com.sun.faces.lifecycle.LifecycleImpl.render(LifecycleImpl.java:159)
 javax.faces.webapp.FacesServlet.service(FacesServlet.java:245)

 

不错,肯定是servlet相关的jar包出了问题,仔细察看Tomcat中的servlet-api.jar和jsp-api.jar包,原来根本就不存在这个所谓的javax/servlet/jsp/tagext/JspIdConsumer接口或类。怎么办?找可用的servlet.jar包!

 

对于JSF1.2应用来说,它使用的JSP API已经用到了新版本,而5.5自带的JSP API明显已经outdate了。众里寻他千百度,总算找到一个JDK5.0的javaee.jar通用包,怎么放呢,如果直接放在common/lib下,两个servlet包必然冲突,删除Tomcat自带的,启动Tomcat的时候会报一堆所谓servlet violate的错误!

 

困惑,迷茫!仔细想想,现在已经早有了Tomcat6.0,为什么不拿过来用用看呢?硬着头皮下载了6.0版本的,将我的应用部署在这个新版本的Tomcat中,哇!一切正常。参考一下6.0根目录下的lib目录,其中jsp-api.jar和servlet-api.jar早已经更新了。

 

如果我事先选择了Tomcat6.0似乎也就不会走这么多弯路,但是,不管怎样,一次经历让自己又多了一份见识,离JSF这位大家闺秀又近了一点,值!

你可能感兴趣的:(java,apache,tomcat,应用服务器,JSF)