vue.js

关注公众号 jb51net

关闭
首页 > 网络编程 > JavaScript > javascript类库 > vue.js > Vue防止重复提交

从入门到精通详解Vue如何实现防止重复提交

作者:大鱼前端

作为经历过大型项目洗礼的前端工程师,小编深知重复提交问题绝非简单的按钮禁用就能解决,今天,就将带大家构建一套生产级的重复提交防御体系,涵盖从基础到架构的全套方案

一、问题本质与解决方案矩阵

在深入代码前,我们需要建立完整的认知框架:

问题维度典型表现解决方案层级
用户行为快速连续点击交互层防御
网络环境请求延迟导致的重复提交网络层防御
业务场景多Tab操作相同资源业务层防御
系统架构分布式请求处理服务端幂等设计

二、基础防御层:用户交互控制

1. 防抖方案

// 适合紧急修复线上问题
const debounceSubmit = (fn, delay = 600) => {
  let timer = null;
  return (...args) => {
    clearTimeout(timer);
    timer = setTimeout(() => fn(...args), delay);
  };
};

适用场景:临时热修复、简单表单

2. 状态变量方案(Vue经典模式)

<template>
  <button 
    @click="handleSubmit"
    :disabled="submitting"
    :class="{ 'opacity-50': submitting }"
  >
    <Spin v-if="submitting" class="mr-1"/>
    {{ submitting ? '提交中...' : '确认提交' }}
  </button>
</template>

<script>
export default {
  data: () => ({
    submitting: false
  }),
  methods: {
    async handleSubmit() {
      if (this.submitting) return;
      
      this.submitting = true;
      try {
        await this.$api.createOrder(this.form);
        this.$message.success('创建成功');
      } finally {
        this.submitting = false;
      }
    }
  }
}
</script>

优化技巧

三、工程化层:可复用方案

1. 高阶函数封装

// utils/submitGuard.js
export const withSubmitGuard = (fn) => {
  let isPending = false;
  
  return async (...args) => {
    if (isPending) {
      throw new Error('请勿重复提交');
    }
    
    isPending = true;
    try {
      return await fn(...args);
    } finally {
      isPending = false;
    }
  };
};

// 使用示例
const guardedSubmit = withSubmitGuard(payload => 
  axios.post('/api/order', payload)
);

2. Vue Mixin方案

// mixins/submitGuard.js
export default {
  data: () => ({
    $_submitGuard: new Set() // 支持多请求并发控制
  }),
  methods: {
    async $guardSubmit(requestKey, fn) {
      if (this.$_submitGuard.has(requestKey)) {
        throw new Error(`[${requestKey}] 请求已在进行中`);
      }
      
      this.$_submitGuard.add(requestKey);
      try {
        return await fn();
      } finally {
        this.$_submitGuard.delete(requestKey);
      }
    }
  }
}

// 组件中使用
await this.$guardSubmit('createOrder', () => (
  this.$api.createOrder(this.form)
));

3. 自定义指令方案(Vue2/Vue3通用)

// directives/v-submit-lock.js
const createSubmitLockDirective = (compiler) => ({
  [compiler === 'vue3' ? 'beforeMount' : 'inserted'](el, binding) {
    const {
      callback,
      loadingText = '处理中...',
      lockClass = 'submit-lock',
      lockAttribute = 'data-submitting'
    } = normalizeOptions(binding);
    
    const originalHTML = el.innerHTML;
    let isSubmitting = false;
    
    const handleClick = async (e) => {
      if (isSubmitting) {
        e.preventDefault();
        e.stopImmediatePropagation();
        return;
      }
      
      isSubmitting = true;
      el.setAttribute(lockAttribute, 'true');
      el.classList.add(lockClass);
      el.innerHTML = loadingText;
      
      try {
        await callback(e);
      } finally {
        isSubmitting = false;
        el.removeAttribute(lockAttribute);
        el.classList.remove(lockClass);
        el.innerHTML = originalHTML;
      }
    };
    
    el._submitLockHandler = handleClick;
    el.addEventListener('click', handleClick, true);
  },
  
  [compiler === 'vue3' ? 'unmounted' : 'unbind'](el) {
    el.removeEventListener('click', el._submitLockHandler);
  }
});

function normalizeOptions(binding) {
  if (typeof binding.value === 'function') {
    return { callback: binding.value };
  }
  
  return {
    callback: binding.value?.handler || binding.value?.callback,
    loadingText: binding.value?.loadingText,
    lockClass: binding.value?.lockClass,
    lockAttribute: binding.value?.lockAttribute
  };
}

// Vue2注册
Vue.directive('submit-lock', createSubmitLockDirective('vue2'));

// Vue3注册
app.directive('submit-lock', createSubmitLockDirective('vue3'));

使用示例

<template>
  <!-- 基础用法 -->
  <button v-submit-lock="handleSubmit">提交</button>
  
  <!-- 配置参数 -->
  <button
    v-submit-lock="{
      handler: submitPayment,
      loadingText: '支付中...',
      lockClass: 'payment-lock'
    }"
    class="btn-pay"
  >
    立即支付
  </button>
  
  <!-- 带事件参数 -->
  <button
    v-submit-lock="(e) => handleSpecialSubmit(e, params)"
  >
    特殊提交
  </button>
</template>

指令优势

4. 组合式API方案(Vue3专属)

// composables/useSubmitLock.ts
import { ref } from 'vue';

export function useSubmitLock() {
  const locks = ref<Set<string>>(new Set());
  
  const withLock = async <T>(
    key: string | symbol,
    fn: () => Promise<T>
  ): Promise<T> => {
    const lockKey = typeof key === 'symbol' ? key.description : key;
    
    if (locks.value.has(lockKey!)) {
      throw new Error(`操作[${String(lockKey)}]已在进行中`);
    }
    
    locks.value.add(lockKey!);
    try {
      return await fn();
    } finally {
      locks.value.delete(lockKey!);
    }
  };
  
  return { withLock };
}

// 组件中使用
const { withLock } = useSubmitLock();

const handleSubmit = async () => {
  await withLock('orderSubmit', async () => {
    await api.submitOrder(form.value);
  });
};

四、架构级方案:指令+拦截器联合作战

1. 智能请求指纹生成

// composables/useSubmitLock.ts
import { ref } from 'vue';

export function useSubmitLock() {
  const locks = ref<Set<string>>(new Set());
  
  const withLock = async <T>(
    key: string | symbol,
    fn: () => Promise<T>
  ): Promise<T> => {
    const lockKey = typeof key === 'symbol' ? key.description : key;
    
    if (locks.value.has(lockKey!)) {
      throw new Error(`操作[${String(lockKey)}]已在进行中`);
    }
    
    locks.value.add(lockKey!);
    try {
      return await fn();
    } finally {
      locks.value.delete(lockKey!);
    }
  };
  
  return { withLock };
}

// 组件中使用
const { withLock } = useSubmitLock();

const handleSubmit = async () => {
  await withLock('orderSubmit', async () => {
    await api.submitOrder(form.value);
  });
};

2. 增强版Axios拦截器

// utils/requestFingerprint.js
import qs from 'qs';
import hash from 'object-hash';

const createFingerprint = (config) => {
  const { method, url, params, data } = config;
  
  const baseKey = `${method.toUpperCase()}|${url}`;
  const paramsKey = params ? qs.stringify(params, { sort: true }) : '';
  const dataKey = data ? hash.sha1(data) : '';
  
  return [baseKey, paramsKey, dataKey].filter(Boolean).join('|');
};

3. 生产级Vue指令(增强版)

// directives/v-request.js
const STATE = {
  IDLE: Symbol('idle'),
  PENDING: Symbol('pending'),
  SUCCESS: Symbol('success'),
  ERROR: Symbol('error')
};

export default {
  beforeMount(el, { value }) {
    const {
      action,
      confirm = null,
      loadingClass = 'request-loading',
      successClass = 'request-success',
      errorClass = 'request-error',
      strategies = {
        duplicate: 'cancel', // cancel|queue|ignore
        error: 'reset' // reset|keep
      }
    } = parseOptions(value);
    
    let state = STATE.IDLE;
    let originalContent = el.innerHTML;
    
    const setState = (newState) => {
      state = newState;
      el.classList.remove(loadingClass, successClass, errorClass);
      
      switch (state) {
        case STATE.PENDING:
          el.classList.add(loadingClass);
          el.disabled = true;
          break;
        case STATE.SUCCESS:
          el.classList.add(successClass);
          el.disabled = false;
          break;
        case STATE.ERROR:
          el.classList.add(errorClass);
          el.disabled = strategies.error === 'keep';
          break;
        default:
          el.disabled = false;
      }
    };
    
    el.addEventListener('click', async (e) => {
      if (state === STATE.PENDING) {
        e.preventDefault();
        return;
      }
      
      try {
        if (confirm && !window.confirm(confirm)) return;
        
        setState(STATE.PENDING);
        await action(e);
        setState(STATE.SUCCESS);
      } catch (err) {
        setState(STATE.ERROR);
        throw err;
      }
    });
  }
};

function parseOptions(value) {
  if (typeof value === 'function') {
    return { action: value };
  }
  
  if (value && typeof value.action === 'function') {
    return value;
  }
  
  throw new Error('Invalid directive options');
}

4. 企业级使用示例

<template>
  <!-- 基础用法 -->
  <button 
    v-request="submitForm"
    class="btn-primary"
  >
    提交订单
  </button>
  
  <!-- 高级配置 -->
  <button
    v-request="{
      action: () => $api.pay(orderId),
      confirm: '确定支付吗?',
      strategies: {
        duplicate: 'queue',
        error: 'keep'
      },
      loadingClass: 'payment-loading',
      successClass: 'payment-success'
    }"
    class="btn-pay"
  >
    <template v-if="$requestState?.isPending">
      <LoadingIcon /> 支付处理中
    </template>
    <template v-else>
      立即支付
    </template>
  </button>
</template>

<script>
export default {
  methods: {
    async submitForm() {
      const resp = await this.$api.submit({
        ...this.form,
        __duplicateStrategy: 'cancel' // 覆盖全局策略
      });
      
      this.$emit('submitted', resp.data);
    }
  }
}
</script>

五、性能与安全增强建议

1.内存优化

2.异常监控

// 在拦截器中添加监控点
const errorInterceptor = (error) => {
  if (axios.isCancel(error)) {
    trackDuplicateRequest(error.message);
  }
  // ...
};

3.服务端协同

// 在请求头添加幂等ID
axios.interceptors.request.use(config => {
  config.headers['X-Idempotency-Key'] = generateIdempotencyKey();
  return config;
});

六、如何选择适合的方案?

写在最后

真正优秀的解决方案需要做到三个平衡:

建议从简单方案开始,随着业务复杂度提升逐步升级防御体系。

以上就是从入门到精通详解Vue如何实现防止重复提交的详细内容,更多关于Vue防止重复提交的资料请关注脚本之家其它相关文章!

您可能感兴趣的文章:
阅读全文