前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >微信小程序踩坑记-Java基于SSM下的post请求

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

作者头像
WindCoder
发布2018-09-19 17:42:17
3.5K0
发布2018-09-19 17:42:17
举报
文章被收录于专栏:WindCoder

前言

最近在持续踩微信小程序的坑,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

解决方案

java后台

方案一与方案二唯一不同之处就是@RequestBody注解的参数类型不同。没错,就是用@RequestBody来获取微信小程序 wx.request中data参数。

方案一:参数为自定义类

该方式主要可用于前端传递自定义对象参数时

代码语言:javascript
复制
    @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这种的,最后无奈放弃选用这种了。

代码语言:javascript
复制
 params" > @ResponseBody
    @RequestMapping("nihao4.do")
    public String nihao4(HttpServletRequest request, HttpServletResponse response, @RequestBody Map<String, Object> 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();
    }

前端

代码语言:javascript
复制
    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接受参数与传递数据的方法

代码语言:javascript
复制
    @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中即可

代码语言:javascript
复制
function json2Form(json) {  
    var str = [];  
    for(var p in json){  
        str.push(encodeURIComponent(p) + "=" + encodeURIComponent(json[p]));  
    }  
    return str.join("&");  
}  
module.exports = {  
  json2Form:json2Form,  
}</span>

使用方式:

使用方式

代码语言:javascript
复制
//获取应用实例  
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解决方案,毕竟翻来覆去就那两项= =)

Spring 注解学习手札(七) 补遗——@ResponseBody,@RequestBody,@PathVariable

angular的post请求,SpringMVC后台接收不到参数值的解决方案(最开始使用了这个的,又是JSON.stringify,又是将给@RequestBodyJSONObjet,然后就报415了。。之后新建了个User实体类并替换了JSONObject后就success了,瞬间幸福感爆棚~)

SpringMVC使用Map或MultiValueMap接收前端提交的Form Data或Query String(找到VO类的解决方案后,想要不需要创建VO类就能获取到信息的方法,然后就遇上了这个,并最终选择了这个用map接收前端参数的方案)

@RequestBody-------springMVC(介个。。我也记不清仔细看了没,看到浏览器打开没关,暂且放着吧)

本文参与 腾讯云自媒体同步曝光计划,分享自作者个人站点/博客。
原始发表:2016-12-14,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • 前言
    • 高危预警
    • 解决方案
      • java后台
        • 方案一:参数为自定义类
        • 方案二:参数为MAP
      • 前端
      • 解决历程
      • 附录
        • HTTP请求中的form data和request payload的区别
          • JSON转Key=val模板
            • 参考资料
            相关产品与服务
            数据库
            云数据库为企业提供了完善的关系型数据库、非关系型数据库、分析型数据库和数据库生态工具。您可以通过产品选择和组合搭建,轻松实现高可靠、高可用性、高性能等数据库需求。云数据库服务也可大幅减少您的运维工作量,更专注于业务发展,让企业一站式享受数据上云及分布式架构的技术红利!
            领券
            问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档