spring MVC中接口参数解析的过程详解
作者:007tangtao 发布时间:2023-11-28 09:17:50
前言
前天工作中遇到了这样一个问题,我在接口的参数封装了一个pojo,这是很常见的,当参数一多,惯性的思维就是封装一个pojo.那么在参数前有很多注解可以添加,比如:@requestParam,@requestBody,@pathvariable等。我的理解是这样的,首先我先申明,我并是没有看过源码,只是凭经验理解。@requestParam试用于get请求,参数在http的header中的URL上,具体放在?后面以key=value的形式存在。@requestBody适用于post请求中参数在http的body中。@pathvariable比较特别是restful的写法,把参数放在URL上,不用问号区分是参数还是URL。也许我这样说不是很准确。但我通常也是这么用的。与此同时,还有一种常见的写法,就是参数前不加注解,比如参数是基本类型的不加@requestParam,参数是bean的不加requestBody,也能被springmvc解析得到。我的接口被小组长看到之后他叫我去掉这个@requestBody,因为后端加上这个之后,前端的ajax请求需要显示的声明content-type:"application/json",才能被springmvc解析得到,这样似乎多做了一件不必要的事情。虽然我按照他的要求去掉了,但是我觉得我得弄清楚这到底是怎么回事,加与不加到底有什么区别,对性能有什么影响,或者各自的最佳适用场景,除了百度,还得问问大神。
spring MVC接口参数解析的过程
首先我自己慢慢的通过debug研究了一下源码。在不添加任何注解的情况下:
在开发的过程中consumes和produces一般都没有加,按道理应该要加上,因为可以减少对接口的查找范围。这是一个简单的demo,我只是需要他来检查springmvc接收请求的流程。
首先在tomcat启动之后,所有controller类中的请求路径也就是@requestMapping随着Controller这个bean加载到了spring的容器中。页面请求过来之后找到DispatcherServlet这个servlet,请求走到servlet之后大家都知道servlet有两种初始化方式,一种是随着立即加载,一种是延迟加载,但是无论怎样,都是只调用一次init方法,然后再以后每次都会直接调用service方法,当tomcat关闭之后servlet的destroy方法被调用生命周期就结束了。所以springmvc是对servlet的封装就必定要继承service方法,DispatcherServlet也就是doDispatch这个方法。这个方法中通过HttpServletRequest对象获得请求路径也就是/notJson,然后与容器中的所有url对比,最终取得Controller中的接口所在。找到了接口自然就知道了接口的参数,我这里就是Display,为了方便简单,Display中只有两个参数,就是下面ajax请求中的两个。
springmvc会通过反射的方式获取到pojo中的属性。在这个过程中首先springmvc会先声明一个数组,这个数组的大小是参数的个数,我这里只有一个,其实我相信很多人会和我遇到相同的问题就是,当参数中同时存在bean和基本类型的参数,springmvc将怎么解析,这个我遇到过几次,在没有看源码的情况下,把基本类型也封装到bean中去了,让前端把属性也写在一个对象中。当然我相信这个不是每个人都能接受的做法,我们都希望搞清楚他究竟是怎样解析的,到时候我们就可以任意摆弄了。下面是反射过程,将我的pojo反射之后获得里面的属性和方法。解析了参数之后,为参数赋值。这里也许是最重要的地方了。究竟是怎么赋值的。
从这个方法debug了解到,name为display,也就是pojo类名的小写,这里不知道为什么springmvc做了这个处理(以后再看)。attribute为带有age和name的对象。不过此时都是null。WebDataBinding用于从Web请求参数到JavaBean对象的数据绑定的特殊DataBinder。接上图bindRequestParameters这个方法,跟进去会发现一个很熟悉的地方就是下图,通过String[] values = request.getParameterValues(paramName);
获得参数名,这个是servlet的获取参数方法,那么就可以知道请求的参数的属性名和属性值。
接下来可想而知就是把这个参数名name换成bean的属性name,参数名age换成属性名age。再跟到这个地方,这个oragina就是上面serclet拿到的属性名值对,把这个map在这转化成PropertyValue。(PropertyValue是用于保存单个bean属性的信息和值的对象。 在此处使用对象,而不是仅将所有属性存储在由属性名称键入的映射中,允许更灵活,并且能够以优化的方式处理索引属性等。请注意,该值不需要是最终所需的类型:BeanWrapper实现应该处理任何必要的转换,因为此对象不知道它将应用于哪些对象。),如此一来就有两个PropertyValue对象了。
转化的时候会忽略不知道的属性
上图是具体转化的方法,方法比较长。下面一句直接给bean赋值。从这个过程来看。只要前端的json对象的属性和后端的bean属性一样,ajax不写content-type,用默认的application/x-www-form-urlencoded; charset=UTF-8
,就能直接赋值。
来源:https://www.cnblogs.com/007tangtao/p/9251861.html


猜你喜欢
- 1、Hutool工具简介HuTool工具(糊涂工具),第三方插件工具,简化操作,是国产的一个产品,界面简洁易懂,比较人性化。(上班可能经常用
- MyBatis中PageHelper不生效今天使用pageHelper,发现设置了PageHelper.startPage(page, pa
- 一、遇到一个问题1、读取CSV文件package com.guor.demo.charset;import java.io.Buffered
- if (textBox1.InvokeRequired) {  
- Android 监听手机GPS打开状态实现代码GPS_Presenterpackage com.yiba.core;import andro
- Java中的BigDecimal类的使用:使用Java中的BigDecimal可以进行精确的计算,但是在使用BigDecimal时我们需要注
- 在Flutter中实现整个App变为灰色是非常简单的,只需要在最外层的控件上包裹ColorFiltered,用法如下:@overrideWi
- 简介上一篇我们讲了简单的动态BroadCast,今天我们通过手工来发送一条BroadCast进一步来了解BroadCast。在上一篇里我们使
- 通过自定义注解的方式(如:@SysLog(obj = "操作对象", text = "操作内容"),
- 本文实例通过Java的Zip输入输出流实现压缩和解压文件,前一部分代码实现获取文件路径,压缩文件名的更改等,具体如下:package com
- 我们在j2ee当中,连接数据库的时候经常会用到properties配置文件,我们原来在eclipse或者myeclipse当中会在src文件
- 前言在我们开发过程中,由于主流的架构都是采用前后端分离的方式,我们作为后端开发者需要为前段持续地提供运行在容器中最新代码,虽然可
- 最近做了关于在Android设备上外接扫码的项目,在此记录一下关于Android USB扫码枪获取内容的问题首先我这边使用是USB HID的
- 简介网上对于 Camera2 的介绍有很多,在 Github 上也有很多关于 Camera2 的封装库,但是对于那些库,封装性太强,有时候我
- 在 Windows 有一些字符是不能作为文件名,尝试重命名一个文件,输入/ 就可以看到windows 提示的不能作为文件名的字符那么具体是包
- 在Java 字符终端上获取输入有三种方式:1、java.lang.System.in (目前JDK版本均支持)2、java.util.Sca
- 本文实例为大家分享了OpenGL绘制贝塞尔曲线的具体代码,供大家参考,具体内容如下最终效果图:通过3个点形成一条贝塞尔曲线1. 鼠标问题在使
- 一、C语言中的变量属性C语言中的变量可以有自己的属性在定义变量的时候可以加上“属性”关键字"
- 本文实例讲述了Java swing框架实现的贪吃蛇游戏。分享给大家供大家参考,具体如下:java是门高级语言,做游戏时适合做后台,但是用它也
- 数据库结构如下strategy中有外键member_id(关联member表)外键strategy_category(关联category表