当请求contenttype为application/x-www-form-urlencoded时通过request.getParameter()可获取到参数值,但是当请求contenttype为其他则获取不到参数值,还有当使用request.getParameter()获取参数之后request.getInputStream()获取内容为空,今天就通过源码来分析一下
首先看下request.getParameter()的源码
其具体实现为org.apache.catalina.connector.Request,此类implements HttpServletRequest
parametersParsed默认为false的一个全局变量,针对当前线程parseParameters()方法只会执行一次,下面看下parseParameters的关键代码
几个关键性的代码
- 当contenttype有多种时,只会取第一种。比如contenttype为application/json;application/x-www-form-urlencoded,最终是application/json
- 当contenttype不为application/x-www-form-urlencoded退出,不在执行下面的关键代码
- 当contenttype为multipart/form-data时,parseParts()方法里使用的解析文件的框架是apache自带的fileupload。
当contenttype为application/x-www-form-urlencoded我们看下下面的关键代码
其中readPostBody方法其实就是request.getinputstream().read();这也是使用过request.getParameter()之后request.getinputstream()为空的原因,因为request.getParameter()调用中已经把request中的inputstream读完了,由于流不能重复读取,所以request.getinputstream()获取就为空了
接着看下面关键代码parameters.processParameters(formData, 0, len);这个方法调用了org.apache.tomcat.util.http.Parameters#processParameters这个方法的工作就是做key和value的拆分,然后将key和value翻到一个变量名为paramHashValues的map中,所以当调用多少次request.getParameter()获取参数都没有问题,因为它已经将请求流中的数据封装到了map中了
总结
- 当contenttype为application/x-www-form-urlencoded时,request.getParameter()才能获取到参数。为其他contenttype时,不执行下面的关键代码,不会读request的输入流
- request.getParameter()可以多次重复使用的原因是此方法将request的输入流数据封装到map中去了,所以可以重复获取
- request.getParameter()和request.getinputstream两个方法之间是互斥的,原因是流是不能重复读取的。而不仅request.getinputstream会读取request的inputstream,request.getParameter()方法内部也会读取request的tinputstream