1、SpringMVC的工作流程?
➢(1)用户发送请求至前端控制器DispatcherServlet;
➢(2)DispatcherServlet收到请求后,调用HandlerMapping处理器映射器,请求获取Handler;
➢(3)处理器映射器根据请求url找到具体的处理器Handler,生成处理器对象及处理器拦截器(如果有则生成),
一并返回给DispatcherServlet;
➢(4)DispatcherServlet 调用 HandlerAdapter处理器适配器,请求执行Handler;
➢(5)HandlerAdapter 经过适配调用 具体处理器进行处理业务逻辑;
➢(6)Handler执行完成返回ModelAndView;
➢(7)HandlerAdapter将Handler执行结果ModelAndView返回给DispatcherServlet;
➢(8)DispatcherServlet将ModelAndView传给ViewResolver视图解析器进行解析;
➢(9)ViewResolver解析后返回具体View;
➢(10)DispatcherServlet对View进行渲染视图(即将模型数据填充至视图中)
➢(11)DispatcherServlet响应用户。
2、Springmvc的优点:
(1)可以支持各种视图技术,而不仅仅局限于JSP;
(2)与Spring框架集成(如IoC容器、AOP等);
(3)清晰的角色分配:
前端控制器(dispatcherServlet)
请求到处理器映射(handlerMapping)
处理器适配器(HandlerAdapter)
视图解析器(ViewResolver)
处理器(Handler)
视图(View)
(4) 支持各种请求资源的映射策略。
3、SpringMVC怎么样设定重定向和转发的?
(1)转发:在返回值前面加 "forward:"
(2)重定向:在返回值前面加 "redirect:"
4、 SpringMVC常用的注解有哪些?
@RequestMapping 用于处理请求 url 映射的注解,可用于类或方法上。
用于类上,则表示类中的所有响应请求的方法都是以该地址作为父路径。
@RequestBody 注解实现接收http请求的json数据,将json转换为java对象。
@ResponseBody 注解实现将controller方法返回对象转化为json对象响应给客户。
@RequestParam 把请求中指定名称的参数给控制器中的形参赋值。
@PathVariable 用于绑定 url 中的占位符。
@RequestHeader 用于获取请求消息头。
@CookieValue 用于把指定 cookie 名称的值传入控制器方法参数。
5、SpingMvc中的控制器的注解一般用哪个?有没有别的注解可以替代?
一般用 @Controller 注解
也可以使用 @RestController ,@RestController 注解相当于 @ResponseBody + @Controller
6、Spring MVC的异常处理 ?
可以将异常抛给Spring框架,由Spring框架来处理;
我们只需要配置简单的异常处理器,在异常处理器中添视图页面即可。
7、SpringMvc的控制器是不是单例模式?如果是,有什么问题?怎么解决?
是单例模式,在多线程访问的时候有线程安全问题,解决方案是在控制器里面不能写可变状态量,
如果需要使用这些可变状态,可以使用ThreadLocal机制解决,为每个线程单独生成一份变量副本,独立操作,互不影响。
8、怎样在方法里面得到Request,或者Response?
在方法的形参中声明(HttpServletRequest request, HttpServletResponse response),
SpringMvc就自动把 request/response 对象传入。
9、SpringMvc中函数的返回值是什么?
返回值可以有很多类型,有String,ModelAndView。ModelAndView类把视图和数据都合并的一起的,但一般用String比较好。
10、SpringMvc用什么对象从后台向前台传递数据的?
通过ModelMap对象,可以在这个对象里面调用put方法,把对象加到里面,前端就可以通过el表达式拿到。
11、怎么样把 ModelMap 里面的数据放入 Session 里面?
可以在类上面加上 @SessionAttributes 注解,里面包含的字符串就是要放入 session 里面的 key。
12、如何自定义拦截器?
➢ 1. 实现拦截器,通常拦截器类可以通过两种方式来定义。
1).通过实现HandlerInterceptor接口,或继承HandlerInterceptor接口的实现类(如HandlerInterceptorAdapter)来定义。
2).通过实现WebRequestInterceptor接口,或继承WebRequestInterceptor接口的实现类来定义。
/**
* 定义拦截器类,实现HandlerInterceptor接口
*
* @author zhao-cj
*/
@Component
public class MyProjectInterceptor implements HandlerInterceptor {
/**
* 原始方法调用前执行的内容
*/
@Override
public boolean preHandle(HttpServletRequest request, HttpServletResponse response,
Object handler) throws Exception {
System.out.println("preHandle...");
return true;
}
/**
* 原始方法调用后执行的内容
*/
@Override
public void postHandle(HttpServletRequest request, HttpServletResponse response,
Object handler, ModelAndView modelAndView) throws Exception {
System.out.println("postHandle...");
}
/**
* 原始方法调用完成后执行的内容
*/
@Override
public void afterCompletion(HttpServletRequest request, HttpServletResponse response,
Object handler, Exception ex) throws Exception {
System.out.println("afterCompletion...");
}
}
➢ 2. 配置拦截器
/**
* SpringMvc 配置器
*
* @author zhao-cj
*/
@Configuration
@EnableWebMvc
public class SpringMvcConfig implements WebMvcConfigurer {
@Autowired
private MyProjectInterceptor projectInterceptor;
private MyProjectInterceptor projectInterceptor2;
/**
* 拦截器配置,可配置多个
*/
@Override
public void addInterceptors(InterceptorRegistry registry) {
registry.addInterceptor(projectInterceptor).addPathPatterns("/mall", "/mall/*");
registry.addInterceptor(projectInterceptor2).addPathPatterns("/mall", "/mall/*");
}
/**
* 静态资源配置
*/
@Override
public void addResourceHandlers(ResourceHandlerRegistry registry) {
registry.addResourceHandler("/resources/**")
.addResourceLocations("classpath:/static/")
.resourceChain(true);
}
/**
* 全局跨域处理
*/
@Override
public void addCorsMappings(CorsRegistry registry) {
//添加映射路径
registry.addMapping("/**")
//是否发送Cookie
.allowCredentials(true)
//设置放行哪些原始域 SpringBoot2.4.4下低版本使用.allowedOrigins("*")
.allowedOriginPatterns("*")
//放行哪些请求方式
.allowedMethods(new String[]{"GET", "POST", "PUT", "DELETE"})
//.allowedMethods("*") //或者放行全部
//放行哪些原始请求头部信息
.allowedHeaders("*")
//暴露哪些原始请求头部信息
.exposedHeaders("*");
}
}