一、Spring MVC 概述
Spring MVC 是 Spring 框架中的一个重要模块,用于构建基于 Java 的 Web 应用程序。它遵循模型-视图-控制器(MVC)设计模式,提供了一种结构化的方式来开发灵活、松耦合的 Web 应用。
Spring MVC 的特点:
1.清晰的分离:严格遵循 MVC 模式,分离业务逻辑、数据和展示层
2.高度可配置:通过 XML 或 Java 注解方式灵活配置
-
强大的注解支持:简化控制器开发
-
无缝集成:与 Spring 其他模块(如 Spring Security、Spring Data)完美集成
-
灵活的视图技术:支持 JSP、Thymeleaf、FreeMarker 等多种视图技术
二、Spring MVC 核心组件
- DispatcherServlet(前端控制器)
作为 Spring MVC 的核心,负责接收所有 HTTP 请求并将它们分发给适当的处理器。
```java
java
// 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-config.xml</param-value>
</init-param>
<load-on-startup>1</load-on-startup>
</servlet>
- 处理器映射(HandlerMapping)
确定哪个控制器应该处理传入的请求。
- 控制器(Controller)
处理请求并返回适当的模型和视图。
java
```java
@Controller
@RequestMapping("/user")
public class UserController {
@GetMapping("/{id}")
public String getUser(@PathVariable Long id, Model model) {
User user = userService.findById(id);
model.addAttribute("user", user);
return "userDetail";
}
}
```
- 视图解析器(ViewResolver)
解析逻辑视图名称到实际视图实现。
java
```java
@Configuration
@EnableWebMvc
public class WebConfig implements WebMvcConfigurer {
@Bean
public ViewResolver viewResolver() {
InternalResourceViewResolver resolver = new InternalResourceViewResolver();
resolver.setPrefix("/WEB-INF/views/");
resolver.setSuffix(".jsp");
return resolver;
}
}
```
三、Spring MVC 请求处理流程
-
请求到达 DispatcherServlet
-
查询 HandlerMapping:确定处理请求的控制器
-
调用控制器方法: 执行业务逻辑
-
返回 ModelAndView: 包含模型数据和视图名称
-
解析视图: 通过 ViewResolver
-
渲染视图: 将模型数据与视图结合
-
返回响应 给客户端
四、常用注解详解
- @RequestMapping
java
```java
@Controller
@RequestMapping("/products")
public class ProductController {
@RequestMapping(method = RequestMethod.GET)
public String listProducts(Model model) {
// 获取产品列表
return "products/list";
}
@RequestMapping(value = "/{id}", method = RequestMethod.GET)
public String getProduct(@PathVariable Long id, Model model) {
// 获取单个产品
return "products/detail";
}
}
- @GetMapping/@PostMapping 等快捷注解
`
java
``java
@GetMapping("/users")
public String userList(Model model) {
// 处理GET请求
}
@PostMapping("/users")
public String createUser(User user) {
// 处理POST请求
}
```
- @RequestParam 和 @PathVariable
java
```java
@GetMapping("/search")
public String search(@RequestParam("keyword") String keyword,
@RequestParam(value = "page", defaultValue = "1") int page) {
// 处理查询参数
}
@GetMapping("/articles/{id}")
public String getArticle(@PathVariable Long id) {
// 处理路径变量
}
```
- @RequestBody 和 @ResponseBody
java
```java
@PostMapping("/api/users")
@ResponseBody
public ResponseEntity<User> createUser(@RequestBody User user) {
// 处理JSON请求体并返回JSON响应
User savedUser = userService.save(user);
return ResponseEntity.ok(savedUser);
}
```
五、数据绑定与表单处理
- 表单绑定示例
java
```java
@Controller
@RequestMapping("/employees")
public class EmployeeController {
@GetMapping("/form")
public String showForm(Model model) {
model.addAttribute("employee", new Employee());
return "employeeForm";
}
@PostMapping("/save")
public String submitForm(@Valid @ModelAttribute("employee") Employee employee,
BindingResult result) {
if (result.hasErrors()) {
return "employeeForm";
}
// 保存员工信息
return "redirect:/employees/list";
}
}
```
- 表单验证
java
```java
public class Employee {
@NotNull
@Size(min=2, max=30)
private String name;
@Min(18)
@Max(60)
private int age;
@Email
private String email;
// getters and setters
}
```
六、异常处理
- @ExceptionHandler
java
```java
@ControllerAdvice
public class GlobalExceptionHandler {
@ExceptionHandler(ResourceNotFoundException.class)
public ResponseEntity<ErrorResponse> handleResourceNotFound(ResourceNotFoundException ex) {
ErrorResponse error = new ErrorResponse("NOT_FOUND", ex.getMessage());
return new ResponseEntity<>(error, HttpStatus.NOT_FOUND);
}
@ExceptionHandler(Exception.class)
public ResponseEntity<ErrorResponse> handleAllExceptions(Exception ex) {
ErrorResponse error = new ErrorResponse("INTERNAL_ERROR", "发生服务器错误");
return new ResponseEntity<>(error, HttpStatus.INTERNAL_SERVER_ERROR);
}
}
```
- @ResponseStatus
java
```java
@ResponseStatus(value = HttpStatus.NOT_FOUND, reason = "资源未找到")
public class ResourceNotFoundException extends RuntimeException {
// 自定义异常
}
```
七、RESTful 服务开发
- REST 控制器示例
java
```java
@RestController
@RequestMapping("/api/users")
public class UserApiController {
@Autowired
private UserService userService;
@GetMapping
public List<User> getAllUsers() {
return 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(@Valid @RequestBody User user) {
User savedUser = userService.save(user);
return ResponseEntity.created(URI.create("/api/users/" + savedUser.getId()))
.body(savedUser);
}
}
```
- 内容协商
Spring MVC 支持根据请求的 Accept 头或扩展名返回不同格式的数据(JSON/XML等)。
java
```java
@GetMapping(value = "/{id}", produces = {MediaType.APPLICATION_JSON_VALUE, MediaType.APPLICATION_XML_VALUE})
public User getUser(@PathVariable Long id) {
return userService.findById(id).orElseThrow(() -> new ResourceNotFoundException("User not found"));
}
```
八、Spring MVC 最佳实践
-
保持控制器简洁:将业务逻辑移到服务层
-
使用适当的HTTP方法:GET用于检索,POST用于创建,PUT用于更新,DELETE用于删除
-
合理使用响应状态码:200 OK, 201 Created, 400 Bad Request等
-
实现全局异常处理:统一处理各种异常
-
使用DTO而非实体:避免直接暴露领域模型
-
添加输入验证:确保数据有效性
-
实现分页和排序:处理大量数据时特别重要
-
考虑使用HATEOAS:使API更易于发现和使用
九、Spring Boot 中的 Spring MVC
Spring Boot 极大简化了 Spring MVC 的配置:
java
```java
@SpringBootApplication
public class MyApplication {
public static void main(String[] args) {
SpringApplication.run(MyApplication.class, args);
}
}
@RestController
@RequestMapping("/api")
public class MyController {
@GetMapping("/hello")
public String hello() {
return "Hello, Spring Boot MVC!";
}
}
自动配置包括:
-
自动注册 DispatcherServlet
-
默认视图解析器
-
静态资源处理
-
默认错误页面
-
消息转换器(JSON/XML)
十、总结
Spring MVC 提供了一个强大而灵活的框架来构建 Web 应用程序。通过其清晰的架构和丰富的功能集,开发者可以快速构建从简单到复杂的企业级应用。掌握 Spring MVC 的核心概念和最佳实践,将帮助您开发出更高效、更易维护的 Web 应用程序。
随着 Spring Boot 的普及,Spring MVC 的开发变得更加简单高效,但底层原理仍然不变。无论是传统的 Web 应用还是现代的 RESTful 服务,Spring MVC 都是一个值得信赖的选择。