美文网首页
接受客户端传递的JSON数据

接受客户端传递的JSON数据

作者: 神豪VS勇士赢 | 来源:发表于2018-08-10 21:00 被阅读235次

POSTMan可以模拟表单,提供请求格式(JSON,非JSON),请求方式。

一 . 非JSON方式:

/**
 * 这种方法可以传值过去
 * 请求非 JSON
 */
@RequestMapping("/getDataSimpleHttpRequest.do")
@ResponseBody
public   Map<String,String> getDataSimpleHttpRequest(HttpServletRequest request){
    Map<String,String> stringStringMap=new HashMap<>();
    stringStringMap.put("username",request.getParameter("username"));
    stringStringMap.put("password",request.getParameter("password"));
    return  stringStringMap;
}
image.png
/**
 * 这种方法不能传值过去
 * 请求非JSON
 */
@RequestMapping("/getDataSimpleMap.do")
@ResponseBody
public  Map<String ,String>  getDataSimpleMap(Map<String ,String> stringStringMap){
    return  stringStringMap;
}
image.png
/**
 * 这种方法可以传值过去
 * 请求非JSON
 */
@RequestMapping("/getDataSimpleDto.do")
@ResponseBody
public User getDataSimpleDto( User user){
    return  user;
}
image.png

方式二 :一半是JSON请求: key=json(了解)
request对象,DTO对象都可以接受json数据
直接Get请求URL中的特殊字符会失败(如:大括号{})
需要URL 编码
可使用 Post 方法提交数据
使用工具POSTMan:GET,POST都可以

/**
 * 这种方法可以传 value 为 json 形式
 * 必须是post
 */
@RequestMapping("/getDataNotJsonNotSimple1.do")
@ResponseBody
public Map<String ,Object> getCar1( User user){
    Map<String ,Object> res = new HashMap<>();
    res.put("username",user.getUsername());
    res.put("password",user.getPassword());
    return res;
}
image.png
/**
 * 这种方法可以传 value 为 json 形式
 * 必须是post
 */
@RequestMapping("/getDataNotJsonNotSimple2.do")
@ResponseBody
public Map<String ,Object> getCar2( HttpServletRequest request){
    Map<String ,Object> res = new HashMap<>();
    res.put("username",request.getParameter("username"));
    res.put("password",request.getParameter("password"));
    return res;
}
image.png

当我们代码使用Map 接收 value 为json类型的时候 就会404


image.png

第二种方式不用指定请求类型,同时value为json格式。、

方式三:纯json请求(掌握)
直接json格式,json前面没有key,可以接受数据的格式如下:

image.png

不再是:url?key1=value1&key2=value2....
必须指定好contentType:我要发送的类型(重点)


image.png

服务端接受使用@RequestBody
@RequestBody接收的是一个Json对象的字符串,而不是一个Json对象。然而在ajax请求往往传的都是Json对象,用JSON.stringify(data)的方式就能将对象变成字符串。同时ajax请求的时候也要指定dataType: "json",contentType:"application/json" 这样就可以将一个对象传到Java端

/**
 *   Content-Type: json 必须设置
 *  此方法也可以传入json对象  前端字段必须对应起来
 */
@RequestMapping("getDataFullJsonMap.do")
@ResponseBody
public  Map<String,Object>  getDataFullJsonMap(@RequestBody Map<String,Object> stringObjectMap){
    Map<String,Object> stringObjectMap1=new HashMap<>();
    String username = (String) stringObjectMap.get("username");
    String  password = (String) stringObjectMap.get("password");
    stringObjectMap1.put("username",username);
    stringObjectMap1.put("password",password);
    return   stringObjectMap1;
}
image.png
/**
 * 此方法可以传入json对象
 */
@RequestMapping("getDataFullJsonDto.do")
@ResponseBody
public  Map<String,Object>   getDataFullJsonDto(@RequestBody User user){
    Map<String,Object> stringObjectMap=new HashMap<>();
    String username = user.getUsername();
    String password = user.getPassword();
    stringObjectMap.put("username",username);
    stringObjectMap.put("password",password);
    return  stringObjectMap;
}
image.png
/**
 * 这种方式 无法直接传入 json 对象
 *这个方法设置 Content-Type: json
 * 并且传入json 对象 之后是无法传入的
 * 只能通过  单独设置请求参数  username  以及  password  才能传入
 */
@RequestMapping("getDataFullJsonHttpRequest.do")
@ResponseBody
public  Map<String,Object>   getDataFullJsonDto1(HttpServletRequest request){
    Map<String,Object> stringObjectMap2=new HashMap<>();
    String username = request.getParameter("username");
    String password = request.getParameter("password");
    stringObjectMap2.put("username",username);
    stringObjectMap2.put("password",password);
    return  stringObjectMap2;
}
image.png

注意:

当我们发送value 为 json 对象的请求时候,会遇到下面的错误:

Note: further occurrences of HTTP header parsing errors will be logged at DEBUG level.

java.lang.IllegalArgumentException: Invalid character found in the request target. The valid characters are defined in RFC 7230 and RFC 3986……

错误原因:

当在浏览器中访问时 URL中带有特殊字符,如花括号冒号时,就会出现这个错误。

例如:http://localhost:8080/index.do?{id:123}

解决方法:
1、去除URL中的特殊字符;

3、使用 Post 方法提交数据

4、更换低版本的Tomcat来规避这种问题。

5、在 conf/catalina.properties 添加或者修改:

5.1 添加 tomcat.util.http.parser.HttpParser.requestTargetAllow=|{}

5.2 修改tomcat/conf/catalina.properties的配置文件
Tomcat在 7.0.73, 8.0.39, 8.5.7 版本后,添加了对于http头的验证。
具体来说,就是添加了些规则去限制HTTP头的规范性
org.apache.tomcat.util.http.parser.HttpParser#IS_NOT_REQUEST_TARGET[]中定义了一堆not request target
if(IS_CONTROL[i] || i > 127 || i == 32 || i == 34 || i == 35 || i == 60 || i == 62 || i == 92 || i == 94 || i == 96 || i == 123 || i == 124 || i == 125) {
IS_NOT_REQUEST_TARGET[i] = true;
}
转换过来就是以下字符(对应10进制ASCII看):
键盘上那些控制键:(<32或者=127)
非英文字符(>127)
空格(32)
双引号(34)

(35)

<(60)

(62)
反斜杠(92)
^(94)
TAB上面那个键,我也不晓得嫩个读(96)
{(123)
}(124)
|(125)

重启服务器后,解决问题。

总结:

个人本地在conf/catalina.properties 中添加 tomcat.util.http.parser.HttpParser.requestTargetAllow=|{} ,成功解决问题。

我同时采用了低版本的 tomcat 也解决了问题 所以大家可以选取其中的一个解决方法。

相关文章

网友评论

      本文标题:接受客户端传递的JSON数据

      本文链接:https://www.haomeiwen.com/subject/ollkbftx.html