Tomcat中Jsp使用没有命名空间类的问题

问题重现:

今天一刚开始学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类,并加载新编译出的Servletclass文件,运行相应的service方法,返回结果给客户端。根据栈信息可以知道问题出在了动态编译Servlet类的时候,而且是说MyDefaultClass无法被解析。

所以我猜测,可能是Tomcat在解析JSP文件的时候,没有发现有用户导入的类型(没有使用命名空间,因而无法导入包,也不需要加包名的前缀,导致Tomcat在解析JSP文件的时候无法自动感知到当前是引用了用户自定义的类的;也导致在编译的时候却能通过),所以在动态编译的时候没有设置ClassPath的值,然后导致了这个问题。然而对于这个“Bug”,Tomcat本身应该不会想不到,何况从逻辑上,默认的把/classes/lib的目录加入到ClassPath中去编译动态生成的Servlet类也是合理的,那么Tomcat为什么要这样做呢?我的猜测可能会有两点吧:1. Tomcat也认为不定义包名是非常不好的习惯,因而故意留下这个缺陷,以惊醒编程人员。2. 也有可能是出于性能的考虑,如果不加ClassPath应该是可以提升编译动态生成的Servlet类的速度的,因而只要不需要ClassPath的情况下,默认就不加。

上一篇:JSP介绍与简单使用


下一篇:d3.js 入门指南