ssm提交post_微信小程序踩坑记-Java基于SSM下的post请求

  • Post author:
  • Post category:java


前言

最近在持续踩微信小程序的坑,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(介个。。我也记不清仔细看了没,看到浏览器打开没关,暂且放着吧)



版权声明:本文为weixin_39775428原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。