问题重现:
今天一刚开始学Java的同学在接触Jsp的时候遇到了一个比较诡异的问题,他在JSP中始终无法使用自己写的类。简单的演示问题代码:
//没有定义包名
public class MyDefaultClass {
}
//index.jsp文件
<%@ page language="java" contentType="text/html; charset=ISO-8859-1"
pageEncoding="ISO-8859-1"%>
<!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd">
<html>
<head>
<title>Insert title here</title>
</head>
<body>
<%
MyDefaultClass cls = new MyDefaultClass();
%>
Load successful
</body>
</html>
出现的错误提示:
type Exception report
message
description The server encountered an internal error () that prevented it from fulfilling this request.
exception
org.apache.jasper.JasperException: Unable to compile class for JSP:
An error occurred at line: 12 in the jsp file: /index.jsp
MyDefaultClass cannot be resolved to a type
9: <body>
10:
11: <%
12: MyDefaultClass cls = new MyDefaultClass();
13: %>
14:
15: Load successful
An error occurred at line: 12 in the jsp file: /index.jsp
MyDefaultClass cannot be resolved to a type
9: <body>
10:
11: <%
12: MyDefaultClass cls = new MyDefaultClass();
13: %>
14:
15: Load successful
Stacktrace:
org.apache.jasper.compiler.DefaultErrorHandler.javacError(DefaultErrorHandler.java:92)
org.apache.jasper.compiler.ErrorDispatcher.javacError(ErrorDispatcher.java:330)
org.apache.jasper.compiler.JDTCompiler.generateClass(JDTCompiler.java:439)
org.apache.jasper.compiler.Compiler.compile(Compiler.java:349)
org.apache.jasper.compiler.Compiler.compile(Compiler.java:327)
org.apache.jasper.compiler.Compiler.compile(Compiler.java:314)
org.apache.jasper.JspCompilationContext.compile(JspCompilationContext.java:592)
org.apache.jasper.servlet.JspServletWrapper.service(JspServletWrapper.java:317)
org.apache.jasper.servlet.JspServlet.serviceJspFile(JspServlet.java:313)
org.apache.jasper.servlet.JspServlet.service(JspServlet.java:260)
javax.servlet.http.HttpServlet.service(HttpServlet.java:717)
解决方案:
为什么要特别提示他是刚开始接触Java呢?因为我是想强调一下他没有定义包名的习惯,所以才会出现这个问题,而在Java中不定义包名是一个非常不好的编程习惯。这个问题本身是可以通过定义自己的包名,然后在JSP文件中导入自己的包解决。
问题原因:
由于JSP文件在请求时动态的生成Servlet类,然后动态的编译新生成的Servlet类,并加载新编译出的Servlet的class文件,运行相应的service方法,返回结果给客户端。根据栈信息可以知道问题出在了动态编译Servlet类的时候,而且是说MyDefaultClass无法被解析。
所以我猜测,可能是Tomcat在解析JSP文件的时候,没有发现有用户导入的类型(没有使用命名空间,因而无法导入包,也不需要加包名的前缀,导致Tomcat在解析JSP文件的时候无法自动感知到当前是引用了用户自定义的类的;也导致在编译的时候却能通过),所以在动态编译的时候没有设置ClassPath的值,然后导致了这个问题。然而对于这个“Bug”,Tomcat本身应该不会想不到,何况从逻辑上,默认的把/classes和/lib的目录加入到ClassPath中去编译动态生成的Servlet类也是合理的,那么Tomcat为什么要这样做呢?我的猜测可能会有两点吧:1. Tomcat也认为不定义包名是非常不好的习惯,因而故意留下这个缺陷,以惊醒编程人员。2. 也有可能是出于性能的考虑,如果不加ClassPath应该是可以提升编译动态生成的Servlet类的速度的,因而只要不需要ClassPath的情况下,默认就不加。