Spring MVC深度解析:从原理到实战

在这里插入图片描述

一、Spring MVC概述

1.1 MVC设计模式

MVC(Model-View-Controller)是一种经典的软件架构模式,将应用程序分为三个核心组件:

  • Model:数据模型,负责业务逻辑和数据管理
  • View:视图层,负责数据展示
  • Controller:控制器,处理用户请求并协调Model和View

1.2 Spring MVC特点

  • 基于DispatcherServlet的前端控制器模式
  • 灵活的配置方式(注解驱动)
  • 强大的数据绑定和验证机制
  • 支持多种视图技术(JSP、Thymeleaf等)
  • 与Spring框架无缝集成

二、Spring MVC核心组件

2.1 架构流程图解

[客户端] --> [DispatcherServlet] 
           --> [HandlerMapping]
           --> [Controller] 
           --> [ModelAndView]
           --> [ViewResolver]
           --> [视图]

2.2 核心组件说明

  • DispatcherServlet:前端控制器,统一处理请求
  • HandlerMapping:请求到处理器的映射
  • Controller:业务逻辑处理器
  • ViewResolver:视图解析器
  • HandlerAdapter:处理器适配器
  • Model:数据模型容器

三、环境搭建与配置

3.1 Maven依赖

<dependencies>
    <!-- Spring MVC -->
    <dependency>
        <groupId>org.springframework</groupId>
        <artifactId>spring-webmvc</artifactId>
        <version>5.3.20</version>
    </dependency>
    
    <!-- Servlet API -->
    <dependency>
        <groupId>javax.servlet</groupId>
        <artifactId>javax.servlet-api</artifactId>
        <version>4.0.1</version>
        <scope>provided</scope>
    </dependency>
</dependencies>

3.2 传统XML配置 vs JavaConfig

XML配置示例:

<!-- web.xml -->
<servlet>
    <servlet-name>dispatcher</servlet-name>
    <servlet-class>
        org.springframework.web.servlet.DispatcherServlet
    </servlet-class>
    <init-param>
        <param-name>contextConfigLocation</param-name>
        <param-value>/WEB-INF/spring-mvc.xml</param-value>
    </init-param>
</servlet>

JavaConfig实现:

@Configuration
@EnableWebMvc
@ComponentScan("com.example.controller")
public class WebConfig implements WebMvcConfigurer {
    
    @Bean
    public ViewResolver viewResolver() {
        InternalResourceViewResolver resolver = new InternalResourceViewResolver();
        resolver.setPrefix("/WEB-INF/views/");
        resolver.setSuffix(".jsp");
        return resolver;
    }
}

四、控制器开发实践

4.1 基础控制器示例

@Controller
@RequestMapping("/user")
public class UserController {

    @GetMapping("/profile")
    public String showProfile(Model model) {
        User user = userService.getCurrentUser();
        model.addAttribute("user", user);
        return "profile";
    }

    @PostMapping("/update")
    public String updateProfile(@Valid UserForm form, 
                              BindingResult result) {
        if (result.hasErrors()) {
            return "edit-profile";
        }
        userService.updateUser(form);
        return "redirect:/user/profile";
    }
}

4.2 请求映射注解

  • @RequestMapping:通用请求映射
  • @GetMapping/@PostMapping:特定HTTP方法映射
  • @PathVariable:URL模板变量
  • @RequestParam:请求参数绑定
@GetMapping("/articles/{id}")
public String getArticle(@PathVariable Long id,
                        @RequestParam(defaultValue = "desc") String sort,
                        Model model) {
    // 业务逻辑
    return "article-detail";
}

在这里插入图片描述

五、数据处理与绑定

5.1 表单处理示例

@Controller
@RequestMapping("/product")
public class ProductController {

    @GetMapping("/create")
    public String showForm(Model model) {
        model.addAttribute("product", new Product());
        return "product-form";
    }

    @PostMapping("/save")
    public String saveProduct(@ModelAttribute("product") Product product,
                             BindingResult result) {
        if (result.hasErrors()) {
            return "product-form";
        }
        productService.save(product);
        return "redirect:/product/list";
    }
}

5.2 数据验证

public class Product {
    
    @NotBlank(message = "产品名称不能为空")
    @Size(max = 50, message = "名称长度不能超过50字符")
    private String name;

    @Min(value = 0, message = "价格不能为负数")
    private BigDecimal price;
    
    // getters/setters
}

六、视图技术集成

6.1 Thymeleaf配置

@Configuration
@EnableWebMvc
public class ThymeleafConfig {

    @Bean
    public SpringResourceTemplateResolver templateResolver() {
        SpringResourceTemplateResolver resolver = new SpringResourceTemplateResolver();
        resolver.setPrefix("/WEB-INF/views/");
        resolver.setSuffix(".html");
        resolver.setTemplateMode("HTML5");
        resolver.setCharacterEncoding("UTF-8");
        return resolver;
    }

    @Bean
    public SpringTemplateEngine templateEngine() {
        SpringTemplateEngine engine = new SpringTemplateEngine();
        engine.setTemplateResolver(templateResolver());
        return engine;
    }
}

6.2 JSP视图示例

<%@ page contentType="text/html;charset=UTF-8" %>
<%@ taglib prefix="c" uri="http://java.sun.com/jsp/jstl/core" %>
<html>
<head>
    <title>用户列表</title>
</head>
<body>
    <h2>用户列表</h2>
    <table>
        <tr>
            <th>ID</th>
            <th>用户名</th>
            <th>邮箱</th>
        </tr>
        <c:forEach items="${users}" var="user">
            <tr>
                <td>${user.id}</td>
                <td>${user.username}</td>
                <td>${user.email}</td>
            </tr>
        </c:forEach>
    </table>
</body>
</html>

七、RESTful API开发

在这里插入图片描述

7.1 REST控制器

@RestController
@RequestMapping("/api/users")
public class UserApiController {

    @Autowired
    private UserService userService;

    @GetMapping
    public ResponseEntity<List<User>> getAllUsers() {
        return ResponseEntity.ok(userService.findAll());
    }

    @GetMapping("/{id}")
    public ResponseEntity<User> getUserById(@PathVariable Long id) {
        return userService.findById(id)
               .map(ResponseEntity::ok)
               .orElse(ResponseEntity.notFound().build());
    }

    @PostMapping
    public ResponseEntity<User> createUser(@RequestBody @Valid User user) {
        User savedUser = userService.save(user);
        return ResponseEntity.created(URI.create("/api/users/" + savedUser.getId()))
                             .body(savedUser);
    }
}

在这里插入图片描述

八、高级特性

在这里插入图片描述

8.1 拦截器实现

public class AuthInterceptor implements HandlerInterceptor {

    @Override
    public boolean preHandle(HttpServletRequest request, 
                            HttpServletResponse response,
                            Object handler) throws Exception {
        if (request.getSession().getAttribute("user") == null) {
            response.sendRedirect("/login");
            return false;
        }
        return true;
    }
}

// 注册拦截器
@Configuration
public class WebConfig implements WebMvcConfigurer {
    
    @Override
    public void addInterceptors(InterceptorRegistry registry) {
        registry.addInterceptor(new AuthInterceptor())
                .addPathPatterns("/**")
                .excludePathPatterns("/login", "/register");
    }
}

8.2 全局异常处理

@ControllerAdvice
public class GlobalExceptionHandler {

    @ExceptionHandler(ResourceNotFoundException.class)
    public ResponseEntity<ErrorResponse> handleNotFound(ResourceNotFoundException ex) {
        ErrorResponse error = new ErrorResponse();
        error.setTimestamp(LocalDateTime.now());
        error.setStatus(HttpStatus.NOT_FOUND.value());
        error.setMessage(ex.getMessage());
        return new ResponseEntity<>(error, HttpStatus.NOT_FOUND);
    }

    @ExceptionHandler(MethodArgumentNotValidException.class)
    public ResponseEntity<ErrorResponse> handleValidationException(
            MethodArgumentNotValidException ex) {
        List<String> errors = ex.getBindingResult()
                .getFieldErrors()
                .stream()
                .map(FieldError::getDefaultMessage)
                .collect(Collectors.toList());

        ErrorResponse error = new ErrorResponse();
        error.setStatus(HttpStatus.BAD_REQUEST.value());
        error.setMessage("Validation failed");
        error.setDetails(errors);
        return new ResponseEntity<>(error, HttpStatus.BAD_REQUEST);
    }
}

九、最佳实践建议

  1. 遵循分层架构原则:

    • Controller层保持精简
    • 业务逻辑放在Service层
    • 数据访问使用Repository模式
  2. 使用DTO进行数据传输:

    public class UserDTO {
        private String username;
        private String email;
        // 省略getter/setter
    }
    
    @PostMapping
    public ResponseEntity<User> createUser(@RequestBody UserDTO dto) {
        User user = userConverter.convertToEntity(dto);
        // ...
    }
    
  3. 性能优化建议:

    • 合理使用缓存(@Cacheable)
    • 启用GZIP压缩
    • 异步处理(@Async)
  4. 安全注意事项:

    • 使用CSRF保护
    • 输入验证和输出编码
    • 参数化查询防止SQL注入

十、常见问题排查

  1. 404错误排查步骤:

    • 检查@RequestMapping注解路径
    • 确认视图解析器配置
    • 查看组件扫描范围
  2. 数据绑定失败处理:

    • 检查字段名称是否匹配
    • 验证数据类型是否兼容
    • 使用@InitBinder进行自定义绑定
  3. 性能问题分析:

    • 使用Spring Actuator监控端点
    • 分析数据库查询性能
    • 检查视图渲染时间
评论 19
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

学编程的小程

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值