前端如何防止接口重复提交

admin 2024-04-21 977 阅读 0评论

什么是接口重复提交?

接口重复提交指的是在网络通信中,同一个请求被客户端多次发送到服务器端的情况。这种情况可能由于多种原因导致,例如用户在等待期间多次点击提交按钮、网络超时后客户端重新发送请求、客户端发送的请求在网络传输过程中出现重复等。

接口重复提交可能会导致多种问题,当服务器收到重复请求时,可能会多次处理相同的数据,导致数据重复操作或者产生不一致的结果。重复提交请求会增加服务器的负载和资源消耗,特别是在高并发情况下,可能会导致服务器压力过大,影响系统的性能和稳定性。有些请求是具有副作用的,例如支付、提交订单等,重复提交可能导致用户被重复扣款或者重复生成订单,从而导致业务异常或者用户不满。

下面我们就来看看前端有哪些防止接口重复提交的方法。

前端如何防止接口的重复提交?

禁用提交按钮

在用户点击提交按钮后,立即禁用该按钮,防止用户多次点击。可以在接口请求结束后重新启用按钮。代码如下所示。

<form id="myForm">
  <!-- 表单内容 -->
  <button type="submit" id="submitButton">提交</button>
</form>

<script>
document.getElementById('myForm').addEventListener('submit', function(event) {
  event.preventDefault(); // 阻止默认提交行为
  document.getElementById('submitButton').disabled = true; // 禁用提交按钮

  // 发送请求
  fetch('/api/submit', {
    method: 'POST',
    // 请求参数
  }).then(function(response) {
    // 处理响应
    document.getElementById('submitButton').disabled = false; // 启用提交按钮
  }).catch(function(error) {
    console.error('Error:', error);
    document.getElementById('submitButton').disabled = false; // 启用提交按钮(如果请求失败)
  });
});
</script>

显示加载状态

在用户提交表单后,显示一个加载状态的提示,告知用户正在处理请求,避免用户重复点击提交按钮。

<form id="myForm">
  <!-- 表单内容 -->
  <button type="submit" id="submitButton">提交</button>
  <div id="loadingMessage" style="display: none;">正在加载...</div>
</form>

<script>
document.getElementById('myForm').addEventListener('submit', function(event) {
  event.preventDefault(); // 阻止默认提交行为
  document.getElementById('submitButton').disabled = true; // 禁用提交按钮
  document.getElementById('loadingMessage').style.display = 'block'; // 显示加载状态

  // 发送请求
  fetch('/api/submit', {
    method: 'POST',
    // 请求参数
  }).then(function(response) {
    // 处理响应
    document.getElementById('submitButton').disabled = false; // 启用提交按钮
    document.getElementById('loadingMessage').style.display = 'none'; // 隐藏加载状态
  }).catch(function(error) {
    console.error('Error:', error);
    document.getElementById('submitButton').disabled = false; // 启用提交按钮(如果请求失败)
    document.getElementById('loadingMessage').style.display = 'none'; // 隐藏加载状态(如果请求失败)
  });
});
</script>

设置防抖或节流

在用户点击提交按钮后,使用防抖或节流的技术延迟发送请求,确保只发送一次请求。防抖和节流是一种常见的前端性能优化技术,可以控制函数的执行频率。

// 防抖函数
function debounce(func, delay) {
  let timer;
  return function() {
    clearTimeout(timer);
    timer = setTimeout(func, delay);
  };
}

document.getElementById('submitButton').addEventListener('click', debounce(function() {
  // 发送请求
  fetch('/api/submit', {
    method: 'POST',
    // 请求参数
  }).then(function(response) {
    // 处理响应
  }).catch(function(error) {
    console.error('Error:', error);
  });
}, 1000)); // 1秒内只允许点击一次

生成请求标识符

在每次请求前生成一个唯一的请求标识符(例如 UUID),并将该标识符作为请求的一部分发送到后端。后端在接收到请求后,检查该标识符是否已经处理过,如果已经处理过则不再处理。前端可以通过记录请求标识符的状态来避免重复提交。

// 生成唯一标识符
function generateRequestId() {
  return Math.random().toString(36).substr(2, 9);
}

let requestId;

document.getElementById('submitButton').addEventListener('click', function() {
  requestId = generateRequestId(); // 生成请求标识符
  // 发送请求
  fetch('/api/submit', {
    method: 'POST',
    headers: {
      'X-Request-Id': requestId // 将请求标识符添加到请求头中
    },
    // 请求参数
  }).then(function(response) {
    // 处理响应
  }).catch(function(error) {
    console.error('Error:', error);
  });
});

使用状态管理库

如果前端使用了状态管理库(如 Redux、Vuex 等),可以在提交请求前检查状态,确保不会重复提交相同的请求。

import store from './store'; // 引入状态管理库

document.getElementById('submitButton').addEventListener('click', function() {
  if (store.state.isSubmitting) {
    return; // 如果正在提交,则不执行后续操作
  }
  // 设置提交状态
  store.commit('setSubmitting', true);

  // 发送请求
  fetch('/api/submit', {
    method: 'POST',
    // 请求参数
  }).then(function(response) {
    // 处理响应
    store.commit('setSubmitting', false); // 恢复非提交状态
  }).catch(function(error) {
    console.error('Error:', error);
    store.commit('setSubmitting', false); // 恢复非提交状态(如果请求失败)
  });
});

接口锁定

在前端发送请求前,先检查是否存在正在处理的相同请求,如果存在则不发送新的请求。可以使用一个变量来记录当前正在处理的请求,以防止重复提交。

let isRequesting = false;

document.getElementById('submitButton').addEventListener('click', function() {
  if (isRequesting) {
    return; // 如果正在请求,则不执行后续操作
  }
  isRequesting = true; // 锁定接口

  // 发送请求
  fetch('/api/submit', {
    method: 'POST',
    // 请求参数
  }).then(function(response) {
    // 处理响应
    isRequesting = false; // 解锁接口
  }).catch(function(error) {
    console.error('Error:', error);
    isRequesting = false; // 解锁接口(如果请求失败)
  });
});

以上方法可以单独使用,也可以组合使用,以提高接口请求的可靠性和安全性。

总结

防止接口重复提交是为了确保系统的数据一致性、避免不必要的资源浪费和提升用户体验。为了避免接口重复提交带来的问题,需要在前端和后端都进行相应的处理,例如在前端禁用提交按钮、显示加载状态,在后端实现幂等性检查等。

喜欢就支持以下吧
点赞 0

发表评论

快捷回复: 表情:
aoman baiyan bishi bizui cahan ciya dabing daku deyi doge fadai fanu fendou ganga guzhang haixiu hanxiao zuohengheng zhuakuang zhouma zhemo zhayanjian zaijian yun youhengheng yiwen yinxian xu xieyanxiao xiaoku xiaojiujie xia wunai wozuimei weixiao weiqu tuosai tu touxiao tiaopi shui se saorao qiudale qinqin qiaoda piezui penxue nanguo liulei liuhan lenghan leiben kun kuaikule ku koubi kelian keai jingya jingxi jingkong jie huaixiao haqian aini OK qiang quantou shengli woshou gouyin baoquan aixin bangbangtang xiaoyanger xigua hexie pijiu lanqiu juhua hecai haobang caidao baojin chi dan kulou shuai shouqiang yangtuo youling
提交
评论列表 (有 0 条评论, 977人围观)

最近发表

热门文章

最新留言

热门推荐

标签列表