getRequestDispatcher("path")在哪里看?

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")正在寻址的内容?

Sot*_*lis 8

Servlet规范解释了这个

getRequestDispatcher方法采用String描述ServletContext范围内的路径的参数.此路径必须相对于ServletContext的根,并以"/"开头,或者为空.该方法使用路径查找servlet,使用第12章"将请求映射到Servlet"中的servlet路径匹配规则,使用 RequestDispatcher对象包装它,并返回结果对象.如果无法根据给定路径解析servlet,则会提供RequestDispatcher,以返回该路径的内容.

这些规则如下

  1. 容器将尝试查找请求路径与servlet路径的完全匹配.成功匹配选择servlet.
  2. 容器将递归地尝试匹配最长的路径前缀.这是通过使用'/'字符作为路径分隔符一次单击目录的路径树来完成的.最长匹配确定所选的servlet.
  3. 如果URL路径中的最后一个段包含扩展名(例如.jsp),则servlet容器将尝试匹配处理扩展请求的servlet.扩展名被定义为最后一个'.'之后的最后一个段的一部分.字符.
  4. 如果前三个规则都不会导致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显示空白页面或显示,请确保您已完成以下操作:

  1. 删除所有addWebapp定义.
  2. addContext然后addWebApp像这样运行所以他们都指向ROOT:

File base = new File("src/com/example/view"); context = tomcat.addContext("", base.getAbsolutePath()); tomcat.addWebapp(null, "/", base.getAbsolutePath());

  1. 在servlet中指向jsp使用request.getRequestDispatcher("/example/read.jsp").forward(request, response);提供的目录/示例存在于"src/com/example/view".