深入浅出Spring MVC

摘要

本文旨在详细分析SpringMVC工作原理以及作为开发者如何基于SpringMVC做扩展。因为SpringMVC分析的文章比较多,所以本文重点讲解如何利用SpringMVC的扩展点实现我们的需求。

什么是Spring MVC

SpringMVC的作用是什么呢?需要解决什么问题呢?

下图是一个客户端与服务端的交互

在这里插入图片描述
在之前的详解http报文(2)-web容器是如何解析http报文的一文中我也提到过。
这次再更细致的分析一遍。一个请求如何中客户端发到服务端,再从服务端返回内容。干的这件事在web中叫请求动态内容,区别于静态内容。在java语言中,为了解决这件事定义了一个规范就是servlet。具体的实现由各大厂商自己定义。

大体部分分为两部分一块是建立连接、一块是传输内容。所以servlet规范包括两大部分,,一部分是servlet接口,定义处理请求的规范。一部分是servlet容器的,去管理加载servlet实例。

轻量级的servlet容器有tomcat/jetty/undertow,servlet框架有SpringMVC/Struts/Webx这些,本篇重点讲解SpringMVC

SpringMVC工作流程

在这里插入图片描述
Spring MVC 顾名思义就是处理Controll-Model-View的。

  1. DispatchServlet 是入口,doDispatch方法开始处理请求
  2. 首先经过controll,controll包含两部分,一部分是url处理映射,将url与具体的处理bean映射起来。也就是HandleMapping,另一部分是具体的Handle,因为需要不同的handle,所以定义了HandleAdapter.
  3. Model比较简单,主要就是ModelView对象,
  4. View 包含两部分,一部分是ViewName的解析,另一部分是ViewName的对应的模板引擎,来渲染出最终的模板引擎。

常见扩展点

基于以上,Spring MVC 提供了不同层面的扩展,方便开发者实现定制化的功能,而不需要底层代码的修改

一. Filter

Filter其实不算是SpringMVC,是servlet的,这时候请求还没有到DispatchServlet。Filter允许对请求和响应做一些统一的定制化处理,比如你限流、日志、trace。

实现javax.servlet.Filter接口即可

二. Controll - HandleMapping,HandlerAdapter

HandleMapping属于Controll层面,我们可以编写任意的HandlerMapping实现类,然后定义策略来决定一个web请求到HandlerExecutionChain对象的生成。

继承RequestMappingHandlerMapping 类即可。
这个具体案例可以看下fredal的博客-使用基于 SpringMVC 的透明 RPC 开发微服务

简要来说,他的rpc通信协议是基于http的。所以rpc调用就是基于服务端还是原来的restful api。写法给普通的前端去掉无异,然后包一层rpc client。方便客户端调用。但是这样太麻烦了,对于不需要暴露给前端的API,单纯是服务间的rpc调用。再走一遍servlet-SpringMVC没必要。

所以他基于RequestMappingHandlerMapping做了一个改造。不再基于SpringMVC,而是自己定义了一套rpc的范式,然后转换为springmvc。

三. Controll - Interceptor

Interceptor属于Controll层面,我们可以自定义各种拦截器,在一个请求被真正处理之前、请求被处理但还没输出到响应中、请求已经被输出到响应中之后这三个时间点去做任何我们想要做的事情。广泛应用于Log,Session,鉴权等场景。

实现HandlerInterceptor接口即可

四. View - HandlerMethodArgumentResolver

解析方法参数的,可以很方便的扩展http请求参数。
实现HandlerMethodArgumentResolver接口即可

比如需要从http header中处理设备信息

@Component
public class DeviceResolver implements HandlerMethodArgumentResolver {
    @Override
    public boolean supportsParameter(final MethodParameter methodParameter) {
        return methodParameter.getParameterType().equals(DeviceInfo.class);
    }

    @Override
    public Object resolveArgument(final MethodParameter methodParameter,
            final ModelAndViewContainer modelAndViewContainer,
            final NativeWebRequest nativeWebRequest,
            final WebDataBinderFactory webDataBinderFactory) throws Exception {
        HttpServletRequest request =
                (HttpServletRequest)  nativeWebRequest.getNativeRequest(HttpServletRequest.class);

        // 从head头中获取设备信息
        String id = request.getHeader("x-device-id");
        if (id != null) {
            DeviceInfo deviceInfo = new DeviceInfo();
            deviceInfo.setId("id");
            return deviceInfo;
        }
        return null;
    }
}

五. View - Converter

类型转换器,主要和序列化相关,参数绑定时springmvc会对将前端传来的参数通过某种规则转化成方法定义的参数的类型,默认实现的有StringHttpMessageConverterByteArrayHttpMessageConverter等等,默认的不能满足需求时我们可自己定义此接口来实现自己的类型的转换。

继承AbstractHttpMessageConverter 即可。

六. View- HandlerExceptionResolver

异常处理,通常需要定义的全局异常。

@ControllerAdvice 注解即可
一次和前端的相互甩锅的问题记录中有总结过这种

七. 修改requestbody 内容

当我们需要对RequestBody的内容进行统一处理时,因为HandlerMethodArgumentResolver只能处理特定类型的,做不到这点要求。

实现RequestBodyAdvice 接口即可。比如我需要处理requestbody中的内容,将emoji输入转换掉

@RestControllerAdvice
public class EmojiReplaceAdvice implements RequestBodyAdvice {
    @Override
    public boolean supports(final MethodParameter methodParameter, final Type targetType,
            final Class<? extends HttpMessageConverter<?>> converterType) {
        return methodParameter.hasParameterAnnotation(EmojiReplace.class);
    }

    @Override
    public Object handleEmptyBody(final Object body, final HttpInputMessage inputMessage,
            final MethodParameter parameter, final Type targetType,
            final Class<? extends HttpMessageConverter<?>> converterType) {
        return body;
    }

    @Override
    public HttpInputMessage beforeBodyRead(final HttpInputMessage inputMessage,
            final MethodParameter parameter,
            final Type targetType, final Class<? extends HttpMessageConverter<?>> converterType)
            throws IOException {
        return new HttpInputMessage() {
            @Override
            public InputStream getBody() throws IOException {
                final String content = IOUtils.toString(inputMessage.getBody());
                final String emojiUnicodeToAlias = StringUtil.parseEmojiUnicodeToAlias(content);
                return new ByteArrayInputStream(
                        emojiUnicodeToAlias.getBytes(StandardCharsets.UTF_8));
            }

            @Override
            public HttpHeaders getHeaders() {
                return inputMessage.getHeaders();
            }
        };
    }

    @Override
    public Object afterBodyRead(final Object body, final HttpInputMessage inputMessage,
            final MethodParameter parameter, final Type targetType,
            final Class<? extends HttpMessageConverter<?>> converterType) {
        return body;
    }
}

总结

这篇文章主要是系统的概括了SpringMVC的工作原理和各种扩展机制,属于高度概括,细节不足。具体的每个扩展点的实现、坑、应用场景需要在之后的文章继续阐述。

参考

https://fredal.xin/develop-with-transparent-rpc

已标记关键词 清除标记
相关推荐
来自一线开fa者的实战经验总结,赠送作者讲解的部分配套视频课程 - 从理论到实践,全面介绍Spring Boot 的原理和应用 - 以高并发场景作为实践案例,循序渐进阐述Spring Boot 实用技巧 - 结合主流持久层框架MyBatis,讲述企业级Spring Boot 开发要点 当前互联网技术盛行,以Spring 框架为主导的Java 互联网技术成了主流,而基于Spring 技术衍生出来的Spring Boot,采用了“约定优于配置”的思想,极大地简化了Spring 框架的开发。随着近些年来微服务技术的流行,Spring Boot 也成了时下炙手可热的热点技术。2017 年9 月,Spring 框架出现了重大版本升级,从4.x 版本升级为了5.x 版本,随着这次升级,Spring Boot的版本也在2018年3月从1.x升级到了2.x。 本书重点讲解Spring Boot 2.x 技术, 首先通过与传统Spring MVC 开发进行对比让读者融入Spring Boot 的开发中,然后再分别介绍全注解下的Spring IoC 和AOP、结合MyBatis 框架详解数据库编程和其事务开发、流行的NoSQL(Redis和MongoDB)技术、Spring MVC 技术、REST风格网站搭建、Spring Security 技术、Spring 5.x 新发布的WebFlux 技术、高并发与锁、打包、测试、监控和Spring Cloud 入门等内容,从而帮助读者掌握企业级Spring Boot 开发的要点。 本书涵盖的内容从理论到互联网微服务后端的实践,无论是Java 互联网后端开发人员,或是从传统Spring 项目转向Spring Boot 的开发人员,还是Java 微服务开发人员,都可以从本书中获益。
©️2020 CSDN 皮肤主题: Age of Ai 设计师:meimeiellie 返回首页