Thr*_*eaT 10 java tomcat servlets
使用嵌入式tomcat,这段代码:
System.out.println("getServletPath: " + request.getServletPath());
System.out.println("getServletContext: " + request.getServletContext().getContextPath());
System.out.println("getServerName: " + request.getServerName());
System.out.println("getServerPort: " + request.getServerPort());
Run Code Online (Sandbox Code Playgroud)
打印出来:
getServletPath: /example
getServletContext:
getServerName: localhost
getServerPort: 9090
Run Code Online (Sandbox Code Playgroud)
这是否意味着:
request.getRequestDispatcher("/example/read.jsp").forward(request, response);
Run Code Online (Sandbox Code Playgroud)
将这个URL forward(request, response)看到JSP:
http://localhost:9090/example/read.jsp?
有没有办法打印出绝对 URL getRequestDispatcher("relativePath")正在寻址的内容?
该Servlet规范解释了这个
该
getRequestDispatcher方法采用String描述ServletContext范围内的路径的参数.此路径必须相对于ServletContext的根,并以"/"开头,或者为空.该方法使用路径查找servlet,使用第12章"将请求映射到Servlet"中的servlet路径匹配规则,使用 RequestDispatcher对象包装它,并返回结果对象.如果无法根据给定路径解析servlet,则会提供RequestDispatcher,以返回该路径的内容.
这些规则如下
- 容器将尝试查找请求路径与servlet路径的完全匹配.成功匹配选择servlet.
- 容器将递归地尝试匹配最长的路径前缀.这是通过使用'/'字符作为路径分隔符一次单击目录的路径树来完成的.最长匹配确定所选的servlet.
- 如果URL路径中的最后一个段包含扩展名(例如.jsp),则servlet容器将尝试匹配处理扩展请求的servlet.扩展名被定义为最后一个'.'之后的最后一个段的一部分.字符.
- 如果前三个规则都不会导致servlet匹配,则容器将尝试提供适合所请求资源的内容.如果为应用程序定义了"默认"servlet,则将使用它.许多容器提供用于提供内容的隐式默认servlet.
你问
这是否意味着:
request.getRequestDispatcher("/ example/display.jsp").forward(request,response); 将查看此URL以转发(请求,响应)到JSP:
http://localhost:9090/example/display.jsp?
不,它不发送HTTP请求,因此该路径与URI无关.它更像是一个内部路径,Servlet容器将尝试与其各种Servlet的url映射匹配.
你也问
有没有办法打印出getRequestDispatcher("relativePath")正在寻址的绝对URL?
不,它并不是一个绝对的URL.它是一个可以由Web应用程序上下文中的某些资源处理的路径.
编辑完成后,您addWebapp将进入Tomcat实例.
tomcat.addWebapp(null, "/view2/example2", new File("src/com/example/view/example").getAbsolutePath());
Run Code Online (Sandbox Code Playgroud)
然后您发送请求
/view2/example2/read.jsp
Run Code Online (Sandbox Code Playgroud)
我打算假设它read.jsp在
src/com/example/view/example/
Run Code Online (Sandbox Code Playgroud)
我相信它位于Web应用程序的可公开访问的部分,因此Servlet容器可以呈现它并使用它进行响应.
您还添加了一个addContext似乎与之类似的Web应用程序addWebapp
context = tomcat.addContext("", base.getAbsolutePath());
Run Code Online (Sandbox Code Playgroud)
并在此上下文中添加了servlet映射.
Tomcat.addServlet(context, "example", new ExampleController());
context.addServletMapping("/example/*", "example");
Run Code Online (Sandbox Code Playgroud)
对于/example/*无法处理,我错了/example.
当您发送请求时
/example
Run Code Online (Sandbox Code Playgroud)
由于上下文路径是"",因此Context将使用上述内容并且映射将与ExampleController上面注册的匹配.您的Servlet代码将执行并到达
request.getRequestDispatcher("/view2/example2/read.jsp").forward(request, response);
Run Code Online (Sandbox Code Playgroud)
注意javadoc ServletRequest#getRequestDispatcher(String)
指定的路径名可能是相对的,但它不能扩展到当前的servlet上下文之外.
换句话说,本Servlet,ExampleController被登记在ServletContext映射到上下文路径"",即.根.路径/view2/example2/read.jsp指的是另一个上下文.由于此上下文没有映射,因此它以404响应.
您可以在不同的上下文中获取对其他Web应用程序的引用.你必须使用ServletContext#getContext(String).例如
ServletContext otherContext = request.getServletContext().getContext("/view2/example2");
Run Code Online (Sandbox Code Playgroud)
现在您拥有了ServletContext,您可以RequestDispatcher在该上下文中获取资源.
otherContext.getRequestDispatcher("/read.jsp").forward(request, response);
Run Code Online (Sandbox Code Playgroud)
因为ServletContext#getRequestDispatcher(String)国家
路径名必须以/开头,并且被解释为相对于当前上下文根.
最终答案:
getRequestDispatcher("path")将在addWebapp引用JSP文件时查看方法中设置的目录.如果NullPointerException显示空白页面或显示,请确保您已完成以下操作:
addWebapp定义.addContext然后addWebApp像这样运行所以他们都指向ROOT:File base = new File("src/com/example/view");
context = tomcat.addContext("", base.getAbsolutePath());
tomcat.addWebapp(null, "/", base.getAbsolutePath());
request.getRequestDispatcher("/example/read.jsp").forward(request, response);提供的目录/示例存在于"src/com/example/view".