当前位置: 首页 > article >正文

在 Spring Boot 的 MVC 框架中 路径匹配的实现 详解

        在 Spring Boot 的 MVC 框架中,路径匹配是由 DispatcherServlet 处理的。它负责将传入的 HTTP 请求映射到合适的控制器方法上。具体来说,Spring MVC 的路径匹配涉及以下几个关键点:


1. 路径匹配的核心流程

  1. DispatcherServlet 接收请求
    当一个 HTTP 请求到达时,Spring Boot 的 DispatcherServlet 负责接收请求,并根据路径将其分发到合适的控制器方法。

  2. HandlerMapping 确定处理器
    Spring MVC 使用 HandlerMapping 接口来查找合适的控制器(@Controller 或 @RestController)以及对应的处理方法(@RequestMapping)。

  3. HandlerAdapter 调用方法
    找到处理器后,HandlerAdapter 会调用相应的方法来处理请求,并返回结果。


2. 路径匹配的注解

Spring MVC 使用注解(主要是 @RequestMapping 及其派生注解)来配置路径匹配规则:

2.1 @RequestMapping 注解

  • 可以用于类和方法级别,用来定义路径。
  • 支持配置路径、HTTP 方法、参数条件等。

示例:

@RestController
@RequestMapping("/api")
public class MyController {

    @RequestMapping("/hello")
    public String sayHello() {
        return "Hello, World!";
    }
}
  • 访问路径为 /api/hello

2.2 组合注解

Spring 提供了一些派生自 @RequestMapping 的组合注解,简化了开发:

注解描述
@GetMapping仅匹配 GET 请求。
@PostMapping仅匹配 POST 请求。
@PutMapping仅匹配 PUT 请求。
@DeleteMapping仅匹配 DELETE 请求。
@PatchMapping仅匹配 PATCH 请求。

示例:

@RestController
@RequestMapping("/users")
public class UserController {

    @GetMapping("/{id}")
    public String getUser(@PathVariable String id) {
        return "User ID: " + id;
    }

    @PostMapping
    public String createUser() {
        return "User created!";
    }
}
  • GET /users/{id} 匹配 getUser 方法。
  • POST /users 匹配 createUser 方法。

2.3 方法参数支持

Spring MVC 的路径匹配支持通过方法参数获取路径中的动态部分:

  1. 路径变量(@PathVariable
    用于从路径中提取动态部分。

    @GetMapping("/users/{id}")
    public String getUser(@PathVariable String id) {
        return "User ID: " + id;
    }
    
  2. 查询参数(@RequestParam
    用于从查询字符串中提取参数。

    @GetMapping("/users")
    public String getUserByQuery(@RequestParam String name) {
        return "User Name: " + name;
    }
    
  3. 请求体(@RequestBody
    用于接收 JSON 或 XML 格式的请求体。

    @PostMapping("/users")
    public String createUser(@RequestBody User user) {
        return "User created: " + user.getName();
    }
    

3. 路径匹配规则

Spring MVC 使用的路径匹配规则比较灵活,支持多种匹配方式:

3.1 精确匹配

路径完全匹配才能调用方法。

示例:

@GetMapping("/hello")
public String hello() {
    return "Hello!";
}
  • 只有请求路径为 /hello 时才会匹配。

3.2 路径参数匹配

可以在路径中定义动态参数,使用 {} 包裹。

示例:

@GetMapping("/users/{id}")
public String getUser(@PathVariable String id) {
    return "User ID: " + id;
}
  • 请求 /users/123 会匹配并返回 User ID: 123

3.3 模糊匹配(通配符 * 和 **

  • * 表示匹配一个路径段(不含 /)。
  • ** 表示匹配任意多个路径段。

示例:

@GetMapping("/files/*")
public String singleLevel() {
    return "Matched single level!";
}

@GetMapping("/files/**")
public String multiLevel() {
    return "Matched multiple levels!";
}
  • 请求 /files/a 匹配 singleLevel
  • 请求 /files/a/b/c 匹配 multiLevel

3.4 带后缀的匹配

可以匹配路径中的后缀,例如 .html 或 .json

示例:

@GetMapping("/docs/{name}.html")
public String getHtmlDoc(@PathVariable String name) {
    return "Document: " + name;
}
  • 请求 /docs/readme.html 会匹配并返回 Document: readme

3.5 正则表达式匹配

可以使用 {变量名:正则表达式} 的形式定义路径。

示例:

@GetMapping("/products/{id:[0-9]+}")
public String getProduct(@PathVariable String id) {
    return "Product ID: " + id;
}
  • 请求 /products/123 匹配。
  • 请求 /products/abc 不匹配。

3.6 默认匹配(优先级)

如果有多个路径同时匹配,请求会按照以下顺序匹配优先级:

  1. 精确匹配。
  2. 带路径参数的匹配。
  3. 带通配符的匹配(* 和 **)。
  4. 默认的 @RequestMapping

4. 路径匹配的注意事项

  1. 大小写敏感
    Spring 默认是大小写敏感的,但可以通过配置关闭:

    spring:
      mvc:
        pathmatch:
          matching-strategy: ant_path_matcher
    
  2. 结尾的斜杠
    路径 /api/test 和 /api/test/ 默认是等价的。

  3. 请求方法不匹配
    如果路径匹配但请求方法不匹配,会返回 405 Method Not Allowed 错误。

  4. 路径冲突
    如果有多个路径规则冲突,Spring 会抛出异常,建议保持路径定义的清晰性和唯一性。


5. Spring Boot 中的路径匹配配置

可以通过以下配置项调整路径匹配行为:

配置示例(application.yml

spring:
  mvc:
    pathmatch:
      matching-strategy: ant_path_matcher  # 匹配策略(默认)
  web:
    pathmatch:
      use-suffix-pattern: false           # 是否启用后缀模式匹配

        Spring Boot 的路径匹配功能强大且灵活,支持多种匹配方式(精确匹配、参数匹配、通配符匹配等)。同时,配合注解(如 @RequestMapping 和 @GetMapping),可以轻松定义 RESTful 风格的接口路径。开发时需要注意路径冲突、匹配优先级以及大小写敏感问题,以确保路径规则清晰且易于维护。


http://www.kler.cn/a/454683.html

相关文章:

  • 低成本Type - C单口便携屏的全方位解析
  • 阿里云新用户服务器配置
  • C 实现植物大战僵尸(一)
  • 变频器在电动机的节能原理是什么?
  • Vim 编辑器详细教程
  • K8s 不同层次的进程间通信实现
  • [python学习笔记]--异常、with、assert
  • 接口Mock技术介绍
  • Exchange ProxyShell 攻击链利用详解
  • 虚拟化 | Proxmox VE 8.x 开源的虚拟化平台快速上手指南
  • 惯性动捕套装与虚拟人应用 | 激活3D虚拟人互动性与表现力
  • Day56 图论part06
  • .net core 的多线程编程
  • stm32f103zet6 i2c 四针 oled 标准库开发
  • 机试题——圣诞节礼盒
  • python监控数据处理应用服务Socket心跳解决方案
  • 树莓集团:解读资源整合是什么意思?
  • C语言基础学习记录-文件管理
  • RabbitMQ工作模式(详解 工作模式:简单队列、工作队列、公平分发以及消息应答和消息持久化)
  • Zettlr(科研笔记) v3.4.1 中文版
  • 在Nginx部署Web应用,如何保障后端API的安全
  • 找到字符串中所有字母异位词
  • 云原生大数据计算服务 MaxCompute
  • IPv6的报头
  • 全视通智慧机构养老方案,含有紧急呼叫、安全保障等系统
  • Lua元方法