1、基于注解的控制器
1.1、@Controller 注解类型
在SpringMVC中使用org.springframework.stereotype.Controller注解类型声明某类的实例是一个控制器。
代码示例:
import org.springframework.stereotype.Controller; @Controller
public class TestController {
}
在SpringMVC中使用扫描机制找到应用中所有基于注解的控制器类,需要在springmvc.xml配置文件中,配置<context:component-scan/>元素指定控制器类的包:
<!--配置扫描控制器类-->
<context:component-scan base-package="com.demo.controller"></context:component-scan>
1.2、@RequestMapping 注解
在基于注解的控制器类中可以为每个请求编写对应的处理方法,需要使用org.springframework.web.bind.annotation.RequestMapping注解类型将请求与处理方法一一对应。
(1)方法级别注解
代码如下:
@Controller
public class TestController {
@RequestMapping("/login")
public String login(){
return "login";
}
}
方法的返回值会自动解析为视图,需要在springmvc.xml中配置视图解析器,并指定前缀和后缀:
<!--配置视图解析器-->
<bean class="org.springframework.web.servlet.view.InternalResourceViewResolver">
<!--配置前缀-->
<property name="prefix" value="/"></property>
<!--配置后缀-->
<property name="suffix" value=".jsp"></property>
</bean>
用户可以使用如下URL方法login()方法:
http://localhost:8080/项目/login
在访问login方法之前需要事先在/WEB-INF/jsp/目录下创建login.jsp文件
(2)类级别的注解
代码如下:
@Controller
@RequestMapping("/index")
public class TestController {
@RequestMapping("/login")
public String login(){
return "login";
}
}
用户可以使用如下URL方法login()方法:
http://localhost:8080/项目/index/login
1.3、请求处理
请求处理方法中常出现的参数类型:
- javax.servlet.http.HttpServletRequest
- javax.servlet.http.HttpServletResponse
- javax.servlet.http.HttpSession
- 输入输出流
- 表单实体类
- 注解类型
- org.springframework.ui.Model
请求处理方法常见的返回类型:
- String类型,表示逻辑视图,需要在springmvc.xml中配置视图解析器;
- ModelAndView类型
2、Controller接收请求参数的常见方式
2.1、通过实体Bean接收请求
通过一个实体Bean来接收请求参数,适用于get和post提交请求方式。需要注意的是,Bean的属性名称必须与请求参数名称相同。
2.2、通过处理方法的形参接收请求
直接把表单参数写到控制器类响应方法的形参中,即形参名称与请求参数名称完全相同,示例代码如下:
jsp页面的form表单
<form action="${pageContext.request.contextPath}/login" method="post">
<p>
用户名:<input type="text" name="uname">
</p>
<p>
密码:<input type="password" name="upwd">
</p>
<p>
<input type="submit" value="登录">
</p>
</form>
controller类接收请求参数
@Controller
public class TestController {
@RequestMapping("/login")
public String login(String uname,String upwd) {
//业务逻辑代码
return "main";
}
}
2.3、通过HttpServletRequest接收请求
jsp页面代码参考本节第2点的form表单,controller类接收请求参数代码如下:
@Controller
public class TestController {
@RequestMapping("/login")
public String login(HttpServletRequest request) {
String uname = request.getParameter("uname");
String upwd = request.getParameter("upwd");
//业务逻辑代码
return "main";
}
}
2.4、通过@PathVariable接收URL中的请求参数
jsp页面代码参考本节第2点的form表单,controller类接收请求参数代码如下:
@Controller
public class TestController {
@RequestMapping("/login/{uname}/{upwd}")
public String login(@PathVariable String uname,@PathVariable String upwd) {
//业务逻辑代码
return "main";
}
}
在访问“http://localhost:8080/login/admin/123456”路径时,上述代码自动将URL中的模板变量{uname}和{upwd}绑定到通过@PathVariable注解到同名参数上,即uname=admin,upwd=123456
2.5、通过@RequestParam接收请求参数
jsp页面代码参考本节第2点的form表单,controller类接收请求参数代码如下:
@Controller
public class TestController {
@RequestMapping("/login")
public String login(@RequestParam String uname, @RequestParam String upwd) {
//业务逻辑代码
return "index";
}
}
该方式与本节第2点中提到的“通过处理方法的形参接收请求参数”的区别是:当请求参数与接收参数名不一致时,“通过处理方法的形参接收请求参数”不会报404错误,而“通过@RequestParam接收请求参数”会报404错误,提示效果如下:
2.6、通过@ModelAttribute接收请求参数
当@ModelAttribute注解放在处理方法的形参上时,用于将多个请求参数封装到一个实体对象,从而简化数据绑定流程,而且自动暴露为模型数据,在视图页面展示时使用。而直接以实体对象作为形参的方式接收请求参数,只是将多个请求参数封装到一个实体对象,并不能暴露为模型数据,需要使用model.addAttribute语句才能暴露为模型数据。
创建实体类,代码示例如下:
public class User {
private String uname;
private String upwd;
//getter和setter方法
}
controller类,代码如下:
@Controller
public class TestController {
@RequestMapping("/login")
public String login(@ModelAttribute("user") User user) {
//业务逻辑代码
return "index";
}
}
3、重定向与转发
在SpringMVC框架中,控制器类中处理方法的return语句默认就是转发实现,只不过实现的是转发到视图,示例代码如下:
@RequestMapping("/login")
public String login() {
//转发到index.jsp
return "index";
}
在SpringMVC框架中,重定向与转发的示例代码如下:
@Controller
@RequestMapping("/index")
public class TestController {
@RequestMapping("/login")
public String login() {
//转发到一个请求方法(同一个控制器类中可以省略/index/)
return "forward:/index/isLogin";
} @RequestMapping("/isLogin")
public String isLogin(){
//重定向到一个请求方法
return "redirect:/index/isRegister";
} @RequestMapping("/isRegister")
public String isRegister(){
//转发到一个视图
return "register";
}
}
在SpringMVC框架中,不管是重定向或转发,都需要符合视图解析器的配置,如果直接转发到一个不需要DispatcherServlet的资源,例如:
//转发到一个静态资源
return "forward:/html/my.html";
则需要在springmvc.xml中配置mvc:resources元素,示例代码:
<mvc:resources mapping="/html/**" location="/html/"></mvc:resources>
4、应用@Autowired进行依赖注入
在Controller类中需要使用org.springframework.beans.factory.annotation.Autowired注解类型将依赖注入到一个属性或方法,例如:
@Autowired
private UserService userService;
在Spring MVC中,为了能被作为依赖注入,类必须使用org.springframework.stereotype.Service注解类型注明为@Service(一个服务)。另外还需要在配置文件中使用<context:component-scan>元素来扫描依赖基础包
Service类代码如下:
import org.springframework.stereotype.Service; @Service
public class UserService {
public void login(User user){
System.out.println(user);
}
}
在springmvc.xml中添加扫描:
<context:component-scan base-package="com.demo.service"></context:component-scan>
Controller类代码如下:
@Controller
public class TestController { @Autowired
private UserService userService; @RequestMapping("/login")
public String login(User user){
userService.login(user);
return "test";
} }
如果在Controller类中注入属性时出错,例如:
出现这种问题的四种原因:
- 如果使用注解配置,service类上可能缺少@Service注解
- 如果用xml配置,是否写了定义
- 检查springmvc.xml中扫描包路径是否正确
- service类中实现的接口是否存在相同方法名的接口
5、@ModelAttribute注解
通过org.springframework.web.bind.annotation.ModelAttribute注解类型可以经常实现以下两个功能:
5.1、绑定请求参数到实体对象
Controller类代码示例:
@Controller
public class TestController {
@RequestMapping("/login")
public String login(@ModelAttribute("user") UserForm user) {
//业务逻辑代码
return "index";
}
}
上述代码中的形参 @ModelAttribute("user") UserForm user 的功能有两个:
- 一是将请求参数的输入封装到user对象中
- 二是创建UserForm实例,以user为键值存储在Model对象中,和 model.addAttribute("user",user) 语句的功能一样。
如果没有指定键值,即使用 @ModelAttribute UserForm user 为形参,那么在创建UserForm实例时以"userForm"为键值存储在Model对象中,和 model.addAttribute("userForm",user) 语句的功能一样。
5.2、注解一个非请求处理方法
被@ModelAttribute注解的方法将在每次调用控制器类的请求处理方法前被调用。这种特性可以用来控制登录权限。
创建BaseController类,代码如下:
public class BaseController {
@ModelAttribute
public void isLogin(HttpSession session) throws Exception {
if(session.getAttribute("user") == null){
throw new Exception("没有权限");
}
}
}
创建接收请求的Controller类,代码如下:
@Controller
@RequestMapping("/admin")
public class TestController extends BaseController {
@RequestMapping("/add")
public String add(){
return "test";
}
}
在访问 "http://localhost:8080/admin/add" 时,会先执行BaseController类中的isLogin()方法,显示效果如下: