Servlet url-pattern /与/*区别,*.action以及SpringMVC中/*解析出错过程分析

以前在使用Servlet的时候,配置url-pattern基本上都是指定的路径,也没有仔细的研究,最近突然发现了一个问题,我们将url-pattern配制成/*,那么Servlet会处理与其匹配的路径,那么我们配制成/是不是效果一样呢?下面我们将通过实际示例来验证一下。
首先我们新建一个Web程序,创建过程这里就不再叙述了。然后我们再创建一个Servlet用于测试

package com.gujin.servlet;

import java.io.IOException;
import java.io.PrintWriter;

import javax.servlet.ServletException;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;

public class UrlPatternTest extends HttpServlet
{
   @Override
   protected void doGet(HttpServletRequest req, HttpServletResponse resp)
         throws ServletException, IOException
   {
      PrintWriter writer = resp.getWriter();
      writer.print("this is UrlPatternTest Servlet.");
      writer.flush();
      writer.close();
   }
}

这个Servlet很简单,只是向页面输出一句话,我们需要在web.xml中进行配置,让其生效。首先我们先将url-pattern配制成/*看一下效果。

<?xml version="1.0" encoding="UTF-8"?>
<web-app xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns="http://java.sun.com/xml/ns/javaee" xsi:schemaLocation="http://java.sun.com/xml/ns/javaee http://java.sun.com/xml/ns/javaee/web-app_2_5.xsd" id="WebApp_ID" version="2.5">
  <display-name>Servlet</display-name>
  <servlet>
    <servlet-name>test</servlet-name>
    <servlet-class>com.gujin.servlet.UrlPatternTest</servlet-class>
  </servlet>
  <servlet-mapping>
    <servlet-name>test</servlet-name>
    <url-pattern>/*</url-pattern>
  </servlet-mapping>
</web-app>

运行程序,通过浏览器访问一下:
我们在浏览器地址栏输入:http://127.0.0.1/Servlet/,查看页面显示结果:
这里写图片描述

Tomcat会把这个请求交给我们刚才创建的Servlet去处理,再输入一些其他的地址也会交给这个Servlet来处理,那么我们是不是可以推断,当url-pattern配置成/*的时候,Tomcat会将所有的请求全部交由对应的Servlet进行处理。当访问的地址正好有对应的文件存在时,依然如此吗?我们来验证一下。
我们继续创建两个文件:test.jsp和test.html

test.html

<!DOCTYPE html>
<html>
<head>
<meta charset="utf-8">
<title>test</title>
</head>
<body>this is test page.
</body>
</html>

test.jsp

<%@ page language="java" contentType="text/html; charset=GB18030"
    pageEncoding="GB18030"%>
<!DOCTYPE html>
<html>
<head>
<meta charset="utf-8">
<title>test</title>
</head>
<body>this is test.jsp page.
</body>
</html>

然后我们通过浏览器访问http://127.0.0.1/Servlet/test.htmlhttp://127.0.0.1/Servlet/test.jsp,我们发现请求依然是交给了我们的Servlet去处理。

下面,修改url-pattern/,我们来验证一下与/*是否相同

web.xml

<?xml version="1.0" encoding="UTF-8"?>
<web-app xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns="http://java.sun.com/xml/ns/javaee" xsi:schemaLocation="http://java.sun.com/xml/ns/javaee http://java.sun.com/xml/ns/javaee/web-app_2_5.xsd" id="WebApp_ID" version="2.5">
  <display-name>Servlet</display-name>
  <servlet>
    <servlet-name>test</servlet-name>
    <servlet-class>com.gujin.servlet.UrlPatternTest</servlet-class>
  </servlet>
  <servlet-mapping>
    <servlet-name>test</servlet-name>
    <url-pattern>/</url-pattern>
  </servlet-mapping>
</web-app>

修改完成后,我们先来访问http://127.0.0.1/Servlet/,结果好像与原来一样,再访问http://127.0.0.1/Servlet/test.html,结果还是一样,我们现在可能会想这两者会不会就没区别呢?别急,我们还有一个地址,好吧,我们最后再访问http://127.0.0.1/Servlet/test.jsp,神奇的事情出现了,结果与原来不一样了。
这里写图片描述
这个时候访问到了真正的jsp页面,而不是由我们的Servlet来处理请求。

最后,我们来总结一下:当url-pattern配置成/*的时候,Tomcat会将所有的请求交给对应的Servlet进行处理,当url-pattern配置成/的时候,多数情况下与/*效果一致,但是,当访问的路径正好对应jsp文件时,Tomcat会访问真实的jsp文件而不是把请求交给对应的Servlet处理。

*.action优先级最低,所以一般不用。但是SpringMVC框架中会用到

个人理解

在SpringMVC中,我们如果把前端控制器的url-pattern配置为<url-pattern>/*</url-pattern>那么就会发生404错误,为什么呢??当我们发送请求到Tomcat服务器的时候,/*匹配时,虽然Tomcat会把我们的请求正确交给我们的Servlet执行,但是当SprinMVC的视图解析器转发jsp视图的时候(看源码知道的),因为url-pattern是/*,所以Tomcat又会拦截我们的请求,把该请求发送给我们配置好的Servlet,导致我们SpringMVC的处理器映射器找不到Handler,最终发生404错误。

    @Override
    protected void renderMergedOutputModel(
            Map<String, Object> model, HttpServletRequest request, HttpServletResponse response) throws Exception {

        // Expose the model object as request attributes.
        exposeModelAsRequestAttributes(model, request);

        // Expose helpers as request attributes, if any.
        exposeHelpers(request);

        // Determine the path for the request dispatcher.
        String dispatcherPath = prepareForRendering(request, response);

        // Obtain a RequestDispatcher for the target resource (typically a JSP).
        RequestDispatcher rd = getRequestDispatcher(request, dispatcherPath);
        if (rd == null) {
            throw new ServletException("Could not get RequestDispatcher for [" + getUrl() +
                    "]: Check that the corresponding file exists within your web application archive!");
        }

        // If already included or response already committed, perform include, else forward.
        if (useInclude(request, response)) {
            response.setContentType(getContentType());
            if (logger.isDebugEnabled()) {
                logger.debug("Including resource [" + getUrl() + "] in InternalResourceView '" + getBeanName() + "'");
            }
            rd.include(request, response);
        }

        else {
            // Note: The forwarded resource is supposed to determine the content type itself.
            if (logger.isDebugEnabled()) {
                logger.debug("Forwarding to resource [" + getUrl() + "] in InternalResourceView '" + getBeanName() + "'");
            }
            rd.forward(request, response);//在这里,视图解析器进行了转发操作。
        }
    }

转载地址

  • 2
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
好的,让我来为你提供一份完整的代码示例。 首先,我们需要创建一个 Maven Web 项目,命名为 student_ms。在项目的 `src/main/webapp` 目录下,创建一个名为 `add_student.jsp` 的 JSP 文件,用于提交学生信息表单。在 `WEB-INF/pages` 目录下,创建一个名为 `student_info.jsp` 的 JSP 文件,用于显示所有学生信息。 接下来,我们需要创建一个名为 `StudentController` 的控制器类,用于接收学生信息并转发到 `student_info.jsp` 页面。该类的代码如下: ```java package com.javaee.ex08.controller; import com.javaee.ex08.model.Student; import org.springframework.stereotype.Controller; import org.springframework.ui.Model; import org.springframework.web.bind.annotation.RequestMapping; import org.springframework.web.bind.annotation.RequestMethod; import java.util.ArrayList; import java.util.List; @Controller public class StudentController { private List<Student> studentList = new ArrayList<>(); @RequestMapping(value = "/addStudent", method = RequestMethod.POST) public String addStudent(Student student, Model model) { studentList.add(student); model.addAttribute("studentList", studentList); return "student_info"; } } ``` 在该类,我们使用 `@Controller` 注解将其声明为一个控制器,使用 `@RequestMapping` 注解将 `addStudent` 方法映射到 `/addStudent` 请求路径。在该方法,我们将提交的学生信息添加到 `studentList` 集合,并将其存储在 `Model` ,然后返回 `student_info` 视图名称,以便显示学生信息。 接下来,我们需要在 `src/main/resources` 目录下创建一个名为 `springmvc-config.xml` 的 Spring MVC 配置文件,用于配置控制器扫描和视图解析器。该文件的代码如下: ```xml <?xml version="1.0" encoding="UTF-8"?> <beans xmlns="http://www.springframework.org/schema/beans" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:mvc="http://www.springframework.org/schema/mvc" xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-4.3.xsd http://www.springframework.org/schema/mvc http://www.springframework.org/schema/mvc/spring-mvc-4.3.xsd"> <mvc:annotation-driven/> <mvc:resources mapping="/resources/**" location="/resources/"/> <bean class="org.springframework.web.servlet.view.InternalResourceViewResolver"> <property name="prefix" value="/WEB-INF/pages/"/> <property name="suffix" value=".jsp"/> </bean> <context:component-scan base-package="com.javaee.ex08.controller"/> </beans> ``` 在该配置文件,我们使用 `<mvc:annotation-driven/>` 标签启用 Spring MVC 注解驱动,使用 `<mvc:resources>` 标签配置静态资源路径。我们还配置了一个 `InternalResourceViewResolver` 视图解析器,用于解析 JSP 视图。最后,我们使用 `<context:component-scan>` 标签扫描 `com.javaee.ex08.controller` 包控制器类。 接下来,在 `web.xml` 文件配置 Spring MVC 前端控制器,代码如下: ```xml <web-app xmlns="http://java.sun.com/xml/ns/javaee" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://java.sun.com/xml/ns/javaee http://java.sun.com/xml/ns/javaee/web-app_3_0.xsd" version="3.0"> <display-name>student_ms</display-name> <servlet> <servlet-name>student_ms</servlet-name> <servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class> <init-param> <param-name>contextConfigLocation</param-name> <param-value>classpath:springmvc-config.xml</param-value> </init-param> <load-on-startup>1</load-on-startup> </servlet> <servlet-mapping> <servlet-name>student_ms</servlet-name> <url-pattern>/</url-pattern> </servlet-mapping> </web-app> ``` 在该配置文件,我们定义了一个名为 `student_ms` 的前端控制器,并将其映射到根路径。我们还在 `init-param` 指定了 `springmvc-config.xml` 配置文件的位置。 最后,我们可以在 `add_student.jsp` 文件创建一个表单,用于提交学生信息。代码如下: ```html <%@ page contentType="text/html;charset=UTF-8" language="java" %> <html> <head> <title>Add Student</title> </head> <body> <h1>Add Student</h1> <form action="${pageContext.request.contextPath}/addStudent" method="post"> <label>学号:</label><input type="text" name="id"><br> <label>姓名:</label><input type="text" name="name"><br> <label>性别:</label> <input type="radio" name="gender" value="male">男 <input type="radio" name="gender" value="female">女<br> <label>年龄:</label><input type="text" name="age"><br> <label>籍贯:</label><input type="text" name="hometown"><br> <label>电话:</label><input type="text" name="phone"><br> <input type="submit" value="提交"> </form> </body> </html> ``` 在该表单,我们使用 `method="post"` 将其提交方法指定为 `POST`,使用 `${pageContext.request.contextPath}` 获取当前应用的上下文路径,将其与 `/addStudent` 拼接成表单的 `action` 属性值。 现在,我们已经完成了一个简单的 Spring MVC Web 项目,可以运行该项目并访问 `add_student.jsp` 页面,提交学生信息并查看 `student_info.jsp` 页面的所有学生信息。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值