在用JSP和Servlet编写Web应用时,经常遇到的问题就是找不到.do路径,或者.do路径不能解析,其实归根到底就是Servlet路径设置的问题,本文将结合实例来进行讲解。
首先,web应用名为MyApp,应用部署容器为Tomcat。
目录结构如下所示:
—MyApp
—WEB-INF/
—classes/
—login.classes
—login.java
—lib/
—web.xml
—index.jsp
—pages/
—login.jsp
web.xml 中配置
<servlet>
<servlet-name>loginhandle</servlet-name>
<servlet-class>/login</servlet-class>
</servlet>
<servlet-mapping>
<servlet-name>loginhandle</servlet-name>
<url-pattern>/login.do</url-pattern>
</servlet-mapping>
服务器端的地址
服务器端的相对地址指的是相对于你的web应用的地址,这个地址是在服务器端解析的(不同于html和javascript中的相对地址,他们是由客户端浏览器解析的)也就是说这时候在jsp和servlet中的相对地址应该是相对于你的web应用,即相对于http://localhost:8080/MyApp/的。
其用到的地方有:
forwarder:servlet中的request.getRequestDispatcher(address);这个address是在服务器端解析的,所以,你要forwarder到login.jsp应该这么写:request.getRequestDispatcher("/pages/login.jsp")这个/相对于当前的web应用webapp,其绝对地址就是:http://localhost:8080/MyApp/pages/login.jsp。
sendRedirect:在jsp中<%response.sendRedirect("/pages/login.jsp");%>
客户端的地址
所有的html中的相对地址都是相对于http://localhost:8080/的,而不是http://localhost:8080/MyApp/的。
Html中的form表单的action属性的地址应该是相对于http://localhost:8080/的,所以,如果提交到login.jsp为:action="/MyApp/pages/login.jsp";提交到servlet为action="/MyApp/login.do"
Javascript也是在客户端解析的,所以其相对路径和form表单一样。
站点根目录和css路径问题
我们称类似这样的相对路径/MyApp/...为相对于站点根目录的相对路径。
当在jsp中引入css时,如果其相对路径相对于当前jsp文件的,而在一个和这个jsp的路径不一样的servlet中forwarder这个jsp时,就会发现这个css样式根本没有起作用。这是因为在servlet中转发时css的路径就是相对于这个servlet的相对路径而非jsp的路径了。所以这时候不能在jsp中用这样的路径:<link href="one.css" rel="stylesheet" type="text/css">或者<link href="../../one.css" rel="stylesheet" type="text/css">。类似href="one.css"和../../one.css的html相对路径是相对于引用这个css的文件的相对路径。而在servlet中转发时就是相对于这个servlet的相对路径了,因为jsp路径和servlet路径是不一样的,所以这样的引用肯定是出错的。
所以这个时候,要用站点根目录,就是相对于http://localhost:8080/的目录,以"/"开头。
因此上述错误应更正为href="/MyApp/one.css"类似的站点根目录的相对目录。这样在servlet转发后和jsp中都是相对于站点根目录的相对路径,就能正确使用所定义的css样式了。
看到这里,相信在路径设置方面应该有一些小成了吧。再看下面这几个实例。
1、 在index.jsp中有<form action="login.do" method="post"></form>,并且该页面的访问地址是http://localhost:8080/MyApp/login.do。此处使用的是相对路径,其中相对路径是相对于当前访问的页面,而当前页面为index.jsp所以当前的上下文路径即为/MyApp。当提交表单后,浏览器地址栏中的信息变为http://localhost:8080/MyApp/login.do,这时候能找到login.do因为已经配置到了web.xml文件中。如果在/pages/login.jsp中包含上面的form表单,则最后的浏览器信息会变为http://localhost:8080/MyApp/pages/login.do。页面肯定会提示错误,因为找不到该路径。
2、在web.xml中的<servlet-mapping>中的<url-pattern>标记中只能使用绝对路径,即必须要以"/"开头。可以这样理解:一个应用对应于一个web.xml,所以这里的绝对路径表示应用的根,而不是服务器的根。(一个特殊情况就是在<url-pattern>中使用模式匹配的*.do,但仍然应该理解成从/开始。)
3、RequestDispatcher view = request.getRequestDispatcher("/pages/login.jsp");
RequestDispatcher view = request.getRequestDispatcher("pages/login.jsp");
这里可以使用相对路径和绝对路径。相对路径是当执行到该代码所在的Servlet的时候地址栏上显示的路径。
如果使用绝对路径,则"/"表示应用的根。
4、RequestDispatcher view = getServletContext().getRequestDispatcher("/pages/login.jsp");
这里必须使用绝对路径,"/"表示应用的根。可以这么理解:在ServletContext上调用请求转发,如果使用相对路径,则无法得知到底相对于该应用的什么位置,只能根据绝对路径来判断。由于是在应用上下文上调用的方法,所以"/"就表示应用的根。
5、servletContext中的getResourceAsStream(String),其中的参数应该是以"/"开头的绝对路径,其中"/"表示应用的根。
如果应用struts框架,那么恭喜你,这些事struts都帮你做了。在struts中,<html:form> 标签自动将action的路径解析为应用路径
总结一下 :
根据处理路径的对象不同将路径分为服务/客户端的路径问题;
通常通过服务器处理的路径称为 相对于服务端的路径: http://localhost:8080/ + 工程名;
通过浏览器处理的路径称为 相对于客户端的路径: http://localhost:8080/ ;
一句话就是他们的相对的根目录不同;
而在具体的使用中,就需要考虑使用相对路径还是绝对路径,有的情况下可以使用两者,有的时候必须使用绝对路径,“/”代表绝对路径也就是代表:http://localhost:8080/;因为在存在页面跳转等情况下,会发生找不到究竟相对于那个目录,就会使用绝对路径;但有时侯必须使用相对路径例如在引用CSS的时候,路径会随着所在页面的不同,而其根目录会不同,就不能使用绝对路径,这时候就应该考虑当前的页面和引用页面之间具体的关系来使用相对路径处理。
web.xml的 / 表示的是http://localhost:8080/ + 工程名 ;
而jsp的action 加了 /的时候表示绝对路径http://localhost:8080/ ;而没有加 / 表示的是相对路径