javaweb项目中有不少场景的路径
客户端的POST/GET请求,服务器的请求转发,资源获取须要设置路径等
这些路径表达的含义都有不一样,因此想要更好的书写规范有用的路径代码 须要对路径有一个清晰地认知java
路径基本上分三类
客户端路径
服务端路径
其余
路径相关的操做类型
超连接
表单
重定向
转发
包含
ServletContext获取资源
Class获取资源
ClassLoader获取资源
客户端路径
超连接/表单/重定向 都是客户端路径
客户端路径形式上能够分为三种方式:
绝对路径;
以“/”开头的相对路径;
不以“/”开头的相对路径;
超连接/表单
好比:
http://127.0.0.1:8080/servlet/NewFile.jsp
绝对路径:连接1以“/”开头的相对路径:连接2相对路径:连接3
绝对路径:
以“/”开头的相对路径:
相对路径:
连接1和表单1:没什么可说的,它使用绝对路径;
连接2和表单2:以“/”开头,相对主机,与当前jsp页面的主机相同
即最终访问的页面为http://localhost:8080/servlet/ServletB;
连接3和表单3:不以“/”开头,相对当前页面的路径,即页面jsp全部的路径
即最终访问的路径为:http://localhost:8080/servlet/ServletB;
重定向
两种形式
带/开头的 相对路径
不带/开头的 相对路径
ServletA中
response.sendRedirect("/servlet/ServletB");
http://127.0.0.1:8080/servlet/ServletA
由于路径以“/”开头,因此相对当前主机
http://127.0.0.1:8080/servlet/ServletB
ServletA中
response.sendRedirect("ServletB");
相对于当前ServletA的路径
当前ServletA的路径为http://127.0.0.1:8080/servlet/
因此请求的是:
http://127.0.0.1:8080/servlet/ServletB
上面的例子若是写成
response.sendRedirect("/ServletB");
将是相对于主机http://127.0.0.1:8080/ServletB 就找不到了
强烈建议使用“/”开头的路径,这说明在页面中的超连接和表单都要以“/”开头
后面是当前应用的名称,再是访问路径
/servlet/ServletB
在Servlet中的重定向也建议使用“/”开头,同理,也要给出应用的名称
应用名称能够使用request.getContextPath()获取
response.sendRedirect(request.getContextPath() + "/BServlet");
服务端路径
请求转发 请求
服务器端路径必须是相对路径,不能是绝对路径。但相对路径有两种形式
以“/”开头;
不以“/”开头;
其中请求转发、请求包含都是服务器端路径
request.getRequestDispatcher()的参数
服务器端路径与客户端路径的区别是:
客户端路径以“/”开头:相对当前主机;
服务器端路径以“/”开头:相对当前应用;
例如
ServletA中
request.getRequestDispatcher("/BServlet").forward(request, response);
相对于ServletA http://127.0.0.1:8080/servlet/ServletA 的当前应用 http://127.0.0.1:8080/servlet
也就是http://127.0.0.1:8080/servlet/BServlet
若是是
request.getRequestDispatcher("BServlet").forward(request, response);
那么则是相对于当前路径
相对于ServletA http://127.0.0.1:8080/servlet/ServletA 的当前路径 http://127.0.0.1:8080/servlet/
也就是http://127.0.0.1:8080/servlet/BServlet
例子不是很明显,由于只有一层路径,应用名下面就是ServletB了 能够再多加一层路径尝试
好比:
假设访问AServlet的路径为:http://localhost:8080/hello/servlet/AServlet
由于路径不以“/”开头,因此相对当前应用,即http://localhost:8080/hello/servlet/BServlet。
其余web
url-pattern
必须使用“/”开头,而且相对的是当前应用。服务器
ServletContext获取资源
必须是相对路径,能够“/”开头,也能够不使用“/”开头,但不管是否使用“/”开头都是相对当前应用路径
String path1 = this.getServletContext().getRealPath("a.txt");
String path2= this.getServletContext().getRealPath("/a.txt");
path1和path2是相同的结果
Class获取资源
importjava.io.InputStream;public classDemo {public voidfun1() {
InputStream in= Demo.class.getResourceAsStream("/a.txt");
}public voidfun2() {
InputStream in= Demo.class.getResourceAsStream("a.txt");
}
}
其中fun1()方法获取资源时以“/”开头,那么相对的是当前类路径,即 /应用名/WEB-INF/classes/a.txt文件;
其中fun2()方法获取资源时没有以“/”开头,那么相对当前Demo.class所在路径,
由于Demo类在com.test 包下,因此资源路径为:/应用名/WEB-INF/classes/com/test/a.txt
Class获取资源也必须是相对路径,能够“/”开头,也能够不使用“/”开头。
以“/”开头,那么相对的是当前类路径,即 /应用名/WEB-INF/classes/
没有以“/”开头,那么相对当前.class所在路径,由于类通常都在包下面好比com.demo
因此/应用名/WEB-INF/classes/...包路径.../
ClassLoader获取资源
ClassLoader获取资源也必须是相对路径,能够“/”开头,也能够不使用“/”开头。
但不管是否以“/”开头,资源都是相对当前类路径。
public classDemo {public voidfun1() {
InputStream in= Demo.class.getClassLoader().getResourceAsStream("/a.txt");
}public voidfun2() {
InputStream in= Demo.class.getClassLoader().getResourceAsStream("a.txt");
}
}
fun1()和fun2()方法的资源都是相对类路径,即classes目录,即/应用名/WEB-INF/classes/a.txt