前言
最近在持续踩微信小程序的坑,canvas和WebSocket的暂时还没找到相关的解决方案,暂时先将post请求无法获取data参数的坑填上。直接附上解决方案,已通过真机检测~之后的解决历程有兴趣的可以看看,没兴趣就可以直接跳过了,附录里是一些遇到的知识延伸以及参考资料地址。本文的环境为:微信小程序+Java+SpringMCV,由于暂时没用到数据库,就不写ssm了。以下内容仅适用于微信小程序,普通页面中暂不负责~(就酱霸气~~)
高危预警
用的久了,发现之前说的那些原因所在了。一切都是Content-Type问题,之前分析的确实有点错误,Content-Type为‘application/json’时,后端需要用@RequestBody ,Content-Type为‘application/x-www-form-urlencoded’或者‘multipart/form-data’时可以用不加注解或者@RequestParam。详情原因可转至最近发布的文章:RequestParam与RequestBod等参数注解简析。
-2017-12-16
内容原因等可能存在误区,暂时没时间深究,请慎重参考。
-2017-08-01
解决方案
方案一与方案二唯一不同之处就是@RequestBody注解的参数类型不同。没错,就是用@RequestBody来获取微信小程序 wx.request中data参数。
方案一:参数为自定义类
该方式主要可用于前端传递自定义对象参数时
@ResponseBody
@RequestMapping(“nihao2.do”)
public String nihao2(HttpServletRequest request, HttpServletResponse response, @RequestBody User j){
response.setHeader(“Access-Control-Allow-Origin”, “*”);
System.out.println(“uid:”+j.getUid()+” uname:”+j.getUname());
JSONObject js = new JSONObject(j);
return js.toString();
}
方案二:参数为MAP
该方法可以自定义前面的传参,原本是想找个能直接传递int、String这种的,最后无奈放弃选用这种了。
params” > @ResponseBody
@RequestMapping(“nihao4.do”)
public String nihao4(HttpServletRequest request, HttpServletResponse response, @RequestBody Map params){
response.setHeader(“Access-Control-Allow-Origin”, “*”);
JSONObject j = new JSONObject(params);
System.out.println(“uid:”+j.get(“uid”)+” uname:”+j.get(“uname”));
return j.toString();
}
前端
wx.request({
url: ‘action地址’,
data: {
‘uid’:1,
‘uname’:”哈哈”
},
method: ‘POST’, // OPTIONS, GET, HEAD, POST, PUT, DELETE, TRACE, CONNECT
// header: {}, // 设置请求的 header
success: function(res){
// success
console.log(“toRed success:”+res.data.uid+’ ‘+res.data.uname)
},
fail: function(res) {
// fail
console.log(“toRed fail:”+res);
},
complete: function(res) {
// complete
console.log(“toRed complete:”+res);
}
})
什么?head怎么没了?data数据转换怎么没了?因为那些都不需要,不论换不换head,最后微信请求时都是’Content-Type’: ‘application/json’,不信的话你可以去开发者工具那看看= =。
解决历程
在网上寻找解决过程时,看到好多说要将head中的’Content-Type’: ‘application/json’改为”Content-Type”: “application/x-www-form-urlencoded”,也有说即使改成上面格式也依旧获取不到data的,然后又将datay由Json改为“uid=1&uname=哈哈”类似字符串,同时字符串要经过encodeURIComponent转码,并给出相应转码util的。然而以上这些都没说后台是什么环境,自然也没找到基于ssm的java后台的相关解决方案了。
偶然转换了下思路,同样的代码为何在普通的html中可以使用,在小程序里就报空指针异常(由于参数中包含int类型)了呢。
常用的SpringMVC接受参数与传递数据的方法
@ResponseBody
@RequestMapping(“nihao.do”)
public String nihao(HttpServletRequest request, HttpServletResponse response, int uid,String uname){
response.setHeader(“Access-Control-Allow-Origin”, “*”);
JSONObject j = new JSONObject();
System.out.println(“uid:”+uid+” uname:”+uname);
j.put(“uid”,uid);
j.put(“uname”,uname);
return j.toString();
}
之后打开前端的开发者工具,发现平常使用ajax传递时,参数数据都在form data中,如图
而微信小程序中的post请求参数是在request payload中,如图:
至于这两者有什么区别,能找到的解释会放在附录中。现在问题就从如何解决微信小程序中post的问题转为用什么方法获取request payload。
经过大量搜索筛选后,@RequestBody这个注解项进入到视野。
@RequestBody 将HTTP请求正文转换为适合的HttpMessageConverter对象。
POST模式下,使用@RequestBody绑定请求对象,Spring会帮你进行协议转换,将Json、Xml协议转换成你需要的对象。
目前能搜到的就是这个了,至于深层的@RequestBody为何能获取到request payload的数据,暂时只能寻求这方面的大神去解答了。在实验过程中,发现它可以用在Map,自定义对象如User(自定义实体类)等上,直接加载JSONobjet发现始终是错误提示415。
附录
HTTP请求中的form data和request payload的区别
整理一下再放送
JSON转Key=val模板
虽然这里用不到,留个备份,没准什么时候用到了呢。
放置在util.js中即可
function json2Form(json) {
var str = [];
for(var p in json){
str.push(encodeURIComponent(p) + “=” + encodeURIComponent(json[p]));
}
return str.join(“&”);
}
module.exports = {
json2Form:json2Form,
}
使用方式:
使用方式
//获取应用实例
var app = getApp()
Page( {
data: {
toastHidden: true,
city_name: ”,
},
onLoad: function() {
var that = this;
wx.request( {
url: “http://op.juhe.cn/onebox/weather/query”,
header: {
“Content-Type”: “application/x-www-form-urlencoded”
},
method: “POST”,
//data: { cityname: “上海”, key: “1430ec127e097e1113259c5e1be1ba70” },
//此处引用
data: Util.json2Form( { cityname: “上海”, key: “1430ec127e097e1113259c5e1be1ba70” }),
complete: function( res ) {
that.setData( {
toastHidden: false,
toastText: res.data.reason,
city_name: res.data.result.data.realtime.city_name,
date: res.data.result.data.realtime.date,
info: res.data.result.data.realtime.weather.info,
});
if( res == null || res.data == null ) {
console.error( ‘网络请求失败’ );
return;
}
}
})
},
onToastChanged: function() {
that.setData( { toastHidden: true });
}
})
var Util = require( ‘../../utils/util.js’ );
参考资料
微信小程序开发之网络请求(POST请求)(放个最全的,基本涵盖了目前网上全部微信小程序post解决方案,毕竟翻来覆去就那两项= =)
angular的post请求,SpringMVC后台接收不到参数值的解决方案(最开始使用了这个的,又是JSON.stringify,又是将给@RequestBodyJSONObjet,然后就报415了。。之后新建了个User实体类并替换了JSONObject后就success了,瞬间幸福感爆棚~)
SpringMVC使用Map或MultiValueMap接收前端提交的Form Data或Query String(找到VO类的解决方案后,想要不需要创建VO类就能获取到信息的方法,然后就遇上了这个,并最终选择了这个用map接收前端参数的方案)
@RequestBody——-springMVC(介个。。我也记不清仔细看了没,看到浏览器打开没关,暂且放着吧)