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;
}
/**
* 这种方法不能传值过去
* 请求非JSON
*/
@RequestMapping("/getDataSimpleMap.do")
@ResponseBody
public Map<String ,String> getDataSimpleMap(Map<String ,String> stringStringMap){
return stringStringMap;
}
/**
* 这种方法可以传值过去
* 请求非JSON
*/
@RequestMapping("/getDataSimpleDto.do")
@ResponseBody
public User getDataSimpleDto( User user){
return user;
}
方式二 :一半是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;
}
/**
* 这种方法可以传 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;
}
当我们代码使用Map 接收 value 为json类型的时候 就会404
第二种方式不用指定请求类型,同时value为json格式。、
方式三:纯json请求(掌握)
直接json格式,json前面没有key,可以接受数据的格式如下:
不再是:url?key1=value1&key2=value2....
必须指定好contentType:我要发送的类型(重点)
服务端接受使用@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;
}
/**
* 此方法可以传入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;
}
/**
* 这种方式 无法直接传入 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;
}
注意:
当我们发送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 也解决了问题 所以大家可以选取其中的一个解决方法。