小程序:前端防止用户重复提交&即时消息(IM)重复发送问题解决
背景:
最近参与开发的小程序,涉及到即时消息(IM)发送的功能;
聊天界面如下,通过键盘上的【发送】按钮,触发消息发送功能
问题发现:
功能开发完毕,进入测试流程;测试工程师反馈说:
在Android手机上,在极短的时间内频繁点击键盘上的【发送】按钮,消息会重复发送;IOS上该问题不太明显
本以为是普通的防重复提交问题,于是自然想到通过设定flag/js加锁的方式解决该问题,于是开始优化代码:
项目基本代码:
wxml:
<input type="text" value="{{msgValue}}" confirm-type="send" bindconfirm="sendMsg" bindinput="bindKeyInput" placeholder="请输入聊天内容" />
JS:
bindKeyInput(e) {
this.setData({
msgValue: e.detail.value.replace(/^\s+|\s+$/g, "")
});
},
sendMsg() {
let self = this;
let msg = self.data.msgValue;
if (msg && self.data.sendMsgState) {
self.data.sendMsgState = false
app.globalData.nim.sendText({
scene: 'p2p',
to: self.data.doctorId,
text: msg,
done(error, msg) {
if (!error) {
//消息发送成功
self.setData({
msgValue: ''
})
} else {
//消息发送失败
wx.showToast({
title: '消息发送失败,请稍后再试',
icon: 'none',
duration: 1500,
mask: true
})
}
}
})
}
}
1# 设定flag/js加锁
//在页面初始数据data中,声明“锁”: sendMsgState data: {
sendMsgState: true
} //在发送消息方法中,符合消息发送条件的时候,把sendMsgState的值置为false;
//并在消息发送成功之后,将消息发送框的value置空的之后,将sendMsgState设为true
sendMsg() {
let self = this;
let msg = self.data.msgValue;
if (msg && self.data.sendMsgState) {
self.data.sendMsgState = false
app.globalData.nim.sendText({
scene: 'p2p',
to: self.data.doctorId,
text: msg,
done(error, msg) {
if (!error) {
//消息发送成功,置空输入框;然后把sendMsgState重新设置为true
self.setData({
msgValue: ''
}, () => {
self.data.sendMsgState = true
})
} else {
//消息发送失败
wx.showToast({
title: '消息发送失败,请稍后再试',
icon: 'none',
duration: 1500,
mask: true
})
}
}
})
}
}
测试结果:
Android手机上依然存在该问题,且很容易复现。
分析原因:
在极短的时间内,频繁点击键盘上的发送按钮;此时:锁(sendMsgState)还没来得及置为false,发送内容输入框的值还没有被清空;
但发送事件已经被有效触发多次,导致了发送消息的重复。
2# 在方案一设定flag/js加锁的基础上,增加连续点击按钮事件间隔少于1s,或者连续两次发送内容相同都停止发送的补充规则
2.1:增加连续点击按钮事件间隔少于1s
经验证:正常的消息发送使用流程,连续两次的消息发送间隔都是超过1s的;间隔小于1s的行为,可判定为重复提交:
具体做法:
步骤一:在data中注册lastSendTime,设置值为空;触发发送事件sendMsg的时候,把当前时间保存到变量currentTime;
步骤二:判断当前时间currentTime与上次发送时间的差值是否小于1000;如果是,则发送事件连续触发时间短于1s,停止发送;
步骤三:消息发送成功之后,在置空内容输入框的setData回调方法中,将lastSendTime的值更新为:currentTime;
2.2:如果当前发送的消息内容和上一次保存在data中的msgValue相同,则可判断连续两次消息重复
因为每次发送成功,data中msg都会被置空;而内容为空的时候,又是不允许发送的;
所以,在短时间内,如果当前发送的消息内容和上一次保存在data中的msgValue相同,则可判断连续两次消息重复
最终优化方案:
sendMsg() {
let self = this;
let msg = self.data.msgValue;
// 防止两次点击操作间隔太快
let currentTime = new Date();
if ((currentTime - this.data.lastSendTime < 1000) || (msg === self.data.msg)) {
//发送事件连续触发时间短于1s,或连续两次发送内容相同,则返回
return;
}
if (msg && self.data.sendMsgState) {
self.data.sendMsgState = false
app.globalData.nim.sendText({
scene: 'p2p',
to: self.data.doctorId,
text: msg,
done(error, msg) {
if (!error) {
self.setData({
msgValue: ''
}, () => {
self.data.sendMsgState = true
self.data.lastSendTime = currentTime
})
} else {
//消息发送失败
wx.showToast({
title: '消息发送失败,请稍后再试',
icon: 'none',
duration: 1500,
mask: true
})
}
}
})
}
}
综上所述:
在单一的flag/js加锁无效的情况下;通过添加额外的规则补充校验,最终方案如下:
在发送内容msg有效及flag/js锁为true的基础上;发送事件sendMsg连续两次触发时间间隔大于或等于1s,及连续两次发送内容不相同的情况下,才允许消息被发送;
最终测试结果:无论是Android,还是IOS都可以正常发送消息,无消息重复发送情况发生了
小程序:前端防止用户重复提交&即时消息(IM)重复发送问题解决的更多相关文章
- 微信小程序,前端大梦想(六)
微信小程序,前端大梦想(六) 微信小程序之联合百度API实现定位 定位功能对于我们都不陌生,在移动端的应用中更是不可或缺的功能,小程序中也提供了对应的API帮助我们完成定位的实现,但是目前小程序的定位 ...
- 1_python小程序之实现用户的注册登陆验证功能
python小程序之实现用户的注册登陆验证功能 程序扼要简述: 一.程序流程:1.程序开始2.判断本地文件/数据库是否已存在用户信息,存在则跳转到登陆,否则跳转到注册,注册成功后后跳转到登陆3.判断 ...
- 关于微信小程序前端Canvas组件教程
关于微信小程序前端Canvas组件教程 微信小程序Canvas接口函数 上述为微信小程序Canvas的内部接口,通过熟练使用Canvas,即可画出较为美观的前端页面.下面是使用微信小程序画图的一些 ...
- 微信小程序前端与myeclipse的数据交换过程(SSH)
这是我个人探究微信小程序前端与后端之间的数据交换的过程,再结合个人所学的SSH框架, 编程工具用myEclipse2014工具.当然,前提是后台的项目要部署到tomcat服务器上才行, 然后总结了从后 ...
- [重要更新]微信小程序登录、用户信息相关接口调整:使用 wx.getUserProfile 取代 wx.getUserInfo
2021年2月24日,微信官方团队发布了一个调整通知:<小程序登录.用户信息相关接口调整说明>,公告明确从4月13日起,所有发布的小程序将无法使用 wx.getUserInfo 接口(JS ...
- 微信小程序之获取用户位置权限(拒绝后提醒)
微信小程序获取用户当前位置有三个方式: 1. wx.getLocation(多与wx.openLocation一起用) 获取当前的精度.纬度.速度.不需要授权.当type设置为gcj02 返回可用于w ...
- 小程序获取微信用户的openid
小程序获取微信用户的openid //index.js //获取应用实例 const app = getApp() Page({ globalData: { appid: '11121221a89e0 ...
- 微信小程序开发——小程序API获取用户位置及异常流处理完整示例
前言: 小程序需要添加一个定位功能,主要的就是获取用户位置的经纬度,然后根据用户经纬度进行一些判断操作. 在小程序提供的Api中,获取用户定位信息的主要Api是 wx.getLocation(obj) ...
- 微信小程序前端页面书写
微信小程序前端页面书写 WXML(WeiXin Markup Language)是框架设计的一套标签语言,结合基础组件.事件系统,可以构建出页面的结构. 一.数据绑定 1. 普通写法 <view ...
随机推荐
- IN、EXISTS的相关子查询用INNER JOIN 代替--性能优化
如果保证子查询没有重复 ,IN.EXISTS的相关子查询可以用INNER JOIN 代替.比如: IN.EXISTS的相关子查询用INNER JOIN 代替--sql2000性能优化
- 【SQL】CASE与DECODE
1. case..when case..when语句用于按照条件返回查询结果,如当我们想把emp表的工资按照多少分成几个不同的级别,并分别统计各个级别的员工数.SQL语句如下: select (cas ...
- Photoshop显示RGB值问题
Bmp与JPEG格式的不同之处在哪里? 使用OpenCV读写图像,然后由Photoshop显示时候结果并不相同,使用jpg格式的图像灰度值明显大于bmp格式,但jpg格式的显示信息是错误的. 过程: ...
- fatal error C1083: 无法打开包括文件:“stdio.h
现象: vs2012一直fatal error C1083: 无法打开包括文件:"stdio.h" 不知道配置太多,动到了什么地方,出现了这个问题: 在: 解决方案--调试源文件 ...
- Android进度条控件ProgressBar使用
ProgressBar有四种样式,圆形的(大,中,小)和直条形的(水平) 对应的style为 <LinearLayout xmlns:android="http://schemas.a ...
- eslint推荐编码规范和airbnb推荐编码规范
Eslint规范 for 循环禁止使用无限循环(这个非默认推荐) // bad for (var i = 0; i < 10; i--) { } for (var i = 10; i >= ...
- 卸载pycharm再重新安装后,找不到第三方库
遇到的问题: 看到pycharm出了新的版本,手痒把旧的版本卸载,然后安装了最新的版本,然后问题就来了. 之前通过PIP命令安装的第三方库,import的时候都报错,找不到模块.既然以前能正常使用,现 ...
- 图的BFS
目录: 一.算法的基本思路 二.算法过程 三.题目:785判断是否为二分图 https://blog.csdn.net/weixin_40953222/article/details/80544928 ...
- 第1章 面向对象的JavaScript
针对基础知识的每一个小点,我都写了一些小例子,https://github.com/huyanluanyu1989/DesignPatterns.git,便于大家理解,如有疑问,大家可留言给我,最近工 ...
- 类型转换、分支(day05)
如果表达式里包含多个不同类型的数字就必须 首先把它们转换成同一个类型然后才能 计算 这个转换过程叫做隐式类型转换,完全由 计算机完成 隐式类型转换过程中一定把占地小的类型转换 成占地大的类型 如果不同 ...