写点什么

跨平台应用开发进阶 (三十六) :uniapp 使用 uni.request 请求报错{“errMsg“:“request:fail abort statusCode:-1“}的解决办法

  • 2022-12-22
    江苏
  • 本文字数:1977 字

    阅读完需:约 6 分钟

跨平台应用开发进阶(三十六) :uniapp使用uni.request请求报错{“errMsg“:“request:fail abort statusCode:-1“}的解决办法

一、问题呈现

uniapp中,使用uni.request()向服务端接口发起请求后,返回值为:{"errMsg":"request:fail abort statusCode:-1"},因此,导致了后面的业务代码报错。


通过检查uni.request()根据服务端接口需要的配置(如:url、data、header、method等)都没有问题。

二、问题分析

2.1 手机没有连网

在以上前提下还出现这种问题,最有可能的原因:手机没有连网,或者看着连上了网但实际网络不可用,可以在手机浏览器中随便输入点文字搜一下测试看看,能不能搜到新内容。


仔细查看,这个{"errMsg":"request:fail abort statusCode:-1 未能找到使用指定主机名的服务器。(-1003)"} 返回值,其实并不是服务端返回来的,因为请求都完全没有到服务端。

2.2 https ssl 证书解析(未得到验证)

通过对比前后版本差异,发现前期后台请求方式为http:IP形式,当前后台请求方式为https:域名形式。通过http:域名形式检测,未发现以上现象。排除法怀疑问题出在 ssl 证书验证阶段。


但是andriod不会出现以上问题,只有ios才会出现以上问题。难道androidios在校验证书方面存在差异?


通过抓包工具 Stream 分析发现,在出现网路连接问题时抓包信息如下:



抓包详情信息如下:


2.3 data 序列化错误

从网上看到如下解释,经过实践,发现问题解决了。


官方文档上说uni.requestcontent-type:application/json时会尝试对data进行序列化。手动将原本是json的数据转成字符串传进去就可以了,希望官方可以对这一块进行优化吧。

2.4 TCP 握手超时失败

通过WhireShark捕获网络层数据,发现 TCP 握手超时。


并在 dmz 区的服务器上捕获 netstat packets rejects in established connections because of timestamp 有这个错误。


通过系统参数优化可解决 TCP 握手超时问题。


在入口服务器打开编辑文件/etc/sysctl.conf,加入以下内容:


net.ipv4.tcp_tw_reuse=0net.ipv4.tcp_tw_recycle=0net.ipv4.tcp_fin_timeout=10net.ipv4.ip_local_port_range=1024 65535net.core.somaxconn=65535vm.drop_caches=1vm.swappiness=0fs.file-max=10485760
复制代码


然后执行 /sbin/sysctl -p 让参数生效。

三、解决办法

3.1 网络检测

针对手机没有连网的场景,在执行uni.request()向服务端请求前,使用uni.getNetworkType()uni.onNetworkStatusChange()对网络进行监听,如果监听到当前设备是处于无网络的状态时,就不执行uni.request()请求,而是弹出一个弹框或者消息提示,提醒用户:“当前设备断网了,请检查网络后重试”。


uni. onNetworkStatusChange 触发时机:在网络切换的时候执行,切换 2G,3G,WiFi 的时候会执行。应用时机:一般都会放在 onShow里面,实时监听网络的切换。当应用关闭后台的时候在 onHide 生命周期函数中执行 uni.offNetworkStatusChange()关闭监听网络。


uni.getNetworkType 触发时机:在页面初始化的时候,或者是在方法执行的时候执行。应用时机:可以再onShowonLoad这样的生命周期里面来执行,也可以在点击事件中添加执行,属于即用即取的。

3.2 请求封装

在使用uni.request进行网络请求时,对入参 data 进行 JSON.stringfy()序列化操作。


return new Promise((resolve, reject) => {  uni.request({    url: curUrl,    method: method || "POST",    header: {      'Content-Type': 'application/json',      'Authorization': 'Basic'    },    // header: headerForm,    data: JSON.stringfy(data) || {},    success(res) {      setToken(res.data)      resolve(res.data)      console.log('The request url(' + curUrl + ')  success return info is: ', res.data)    },    fail(err) {      reject(err)      console.log('The request url(' + curUrl + ')  fail return info is: ', err)    },    complete() {      // uni.hideLoading()    }  })}).catch(err => {  handleErr(err)})
复制代码


官方文档中关于 data 数据说明信息如下:


最终发送给服务器的数据是 String 类型,如果传入的 data 不是 String 类型,会被转换成String。转换规则如下:

  • 对于 GET 方法,会将数据转换为 query string。例如 { name: 'name', age: 18 } 转换后的结果是 name=name&age=18

  • 对于 POST 方法且 header['content-type']application/json 的数据,会进行 JSON 序列化。

  • 对于 POST 方法且 header['content-type']application/x-www-form-urlencoded 的数据,会将数据转换为 query string

四、拓展阅读

发布于: 刚刚阅读数: 3
用户头像

No Silver Bullet 2021-07-09 加入

岂曰无衣 与子同袍

评论

发布
暂无评论
跨平台应用开发进阶(三十六) :uniapp使用uni.request请求报错{“errMsg“:“request:fail abort statusCode:-1“}的解决办法_uni-app_No Silver Bullet_InfoQ写作社区