• request.getParameter()、request.getInputStream()和request.getReader()


    这是一个异常引发的故事:

    需要实现一个手机客户端行为分析的需求,手机客户端需要上传一些数据文件。手机客户端通过http协议post方式上传数据文件的时候,我们发现在服务器端无法通过request.getInputStream()获取到相应的数据,调用request.getInputStream()读取数据里头啥都木有。


    经过各种尝试我们注意到如下的情况:

    1.在不做任何修改的情况下,调用request.getParameter()可以获取到部分数据,即url拼接参数的数据。

    2.在不做任何修改的情况下,如果在之前任何地方都不调用request.getParameter(),而是直接调用request.getInputStream()是可以获取到数据的。

    3.原来客户端使用默认的请求头Content-Type:application/x-www-form-urlencoded,修改此值为multipart/form-data或者application/octet-stream之后,通过request.getInputStream()可以获取到数据,即使之前调用过request.getParameter(),最后的解决方法当然是客户端修改Content-Type。

    一.form的enctype属性与Content-Type

    写html的时候我们都知道form有个属性enctype,默认值是application/x-www-form-urlencoded,这个值表示会将表单数据用&符号做一个简单的拼接。例如:

    [plain] view plaincopy
    1. POST /post_test.php HTTP/1.1   
    2. Accept-Language: zh-CN  
    3. User-Agent: Mozilla/4.0   
    4. Content-Type: application/x-www-form-urlencoded   
    5. Host: 192.168.12.102  
    6. Content-Length: 42  
    7. Connection: Keep-Alive  
    8. Cache-Control: no-cache  
    9.    
    10. title=test&content=%B3%AC%BC%B6%C5%AE%C9%FA&submit=post+article  
    我们注意到这个时候Content-Type为application/x-www-form-urlencoded。

    如果enctype的值为multipart/form-data,这个值一般用于表单中包含文件上传的情况,它会将表单中的数据使用一个boundary作为分隔上传。例如:

    [plain] view plaincopy
    1. POST /post_test.php?t=1 HTTP/1.1  
    2. Accept-Language: zh-CN  
    3. User-Agent: Mozilla/4.0    
    4. Content-Type: multipart/form-data; boundary=---------------------------7dbf514701e8  
    5. Accept-Encoding: gzip, deflate  
    6. Host: 192.168.12.102  
    7. Content-Length: 345  
    8. Connection: Keep-Alive  
    9. Cache-Control: no-cache  
    10.    
    11. -----------------------------7dbf514701e8  
    12. Content-Disposition: form-data; name="title"  
    13. test  
    14. -----------------------------7dbf514701e8  
    15. Content-Disposition: form-data; name="content"  
    16. ....  
    17. -----------------------------7dbf514701e8  
    18. Content-Disposition: form-data; name="submit"  
    19. post article  
    20. -----------------------------7dbf514701e8--  
    我们注意到这个时候Content-Type也相应的变为multipart/form-data,同时后面还加上了分隔符boundary的描述。

    所以,其实form的enctype属性某种程度上决定了Content-Type值和请求body里头的数据格式。


    最后注意 request.getParameter()、 request.getInputStream()、request.getReader()这三种方法是有冲突的,因为流只能被读一次。例如: 

    1. 当form表单内容采用 enctype=application/x-www-form-urlencoded编码时,先通过调用request.getParameter() 方法得到参数后,再调用 request.getInputStream()或request.getReader()已经得不到流中的内容,因为在调用 request.getParameter()时系统可能对表单中提交的数 据以流的形式读了一次,反之亦然。 
    2. 当form表单内容采用 enctype=multipart/form-data编码时,即使先调用request.getParameter()也得不到数据,但是这时调用 request.getParameter()方法对 request.getInputStream()或request.getReader()没有冲突,即使已经调用了 request.getParameter()方法也 可以通过调用request.getInputStream()或request.getReader()得 到表单中的数据,而request.getInputStream()和request.getReader()在同 一个响应中是不能混合使用的,如果混合使用就会抛异常。 

  • 相关阅读:
    Cannot assign requested address问题总结
    Trying to connect an http1.x server
    从 0 到 1 搭建技术中台之推送平台实践:高吞吐、低延迟、多业务隔离的设计与实现
    思考gRPC :为什么是HTTP/2
    HTTP/1HTTP/2HTTP/3
    get_or_create update_or_create
    死锁案例 GAP 锁 没有就插入,存在就更新
    死锁产生必要条件
    京东零售mockRpc实践
    Certbot CA 证书 https
  • 原文地址:https://www.cnblogs.com/ycpanda/p/3637209.html
Copyright © 2020-2023  润新知