首页 > 其他分享 >重复提交数据造成脏数据解决方案

重复提交数据造成脏数据解决方案

时间:2023-11-06 09:24:24浏览次数:37  
标签:return 数据 解决方案 data 提交 error const message config

1、 前端请求限制重复,vue代码如下

import axios from "~../../axios"
import {Loading, Message, MessageBox, Notification} from "~../../element-ui"
import store from "~../../../src/store"
import {getToken} from "./auth"
import errorCode from "./errorCode"
import {blobValidate, tansParams} from "./ruoyi"
import {saveAs} from "~../../file-saver"
import {getAbsolutePath} from "./index"

let downloadLoadingInstance

// 是否显示重新登录
let isReloginShow

// 请求重复提交限制 2023-11-03
const requestQueue = {
    //请求限制时间,单位毫秒;0不启用,大于0启用;一秒钟不允许重复提交
    limitTime: 1000,
    //存放请求配置队列
    queue: [
        // {
        //     createTime: Number,
        //     url: String,
        //     method: String,
        //     cancel: Function,
        //     dataMd5: String
        // }
    ],
    //请求限制时间的方法
    limitMethods: ['post', 'delete', 'put'],
    //当前请求是否可用
    canUsed(config) {
        if (isNaN(this.limitTime) || Number(this.limitTime) <= 0) return false;
        //只验证post提交数据
        if (config) {
            let method = (config.method || '').toLowerCase();
            return this.limitMethods.indexOf(method) > -1;
        }
        return true;
    },
    isSameRequest(r1, r2) {
        let same = r1.url === r2.url && r1.method === r2.method;
        if (r1.dataMd5 || r2.dataMd5) {
            same = same && r1.dataMd5 == r2.dataMd5
        }
        return same;
    },
    //获取数据的md5值,暂未实现,根据后续需要决定是否实现
    createDataMd5(data) {
        return data ? null : null;
    },
    //添加请求到队列
    addRequest(config) {
        if (!this.canUsed(config)) return;
        let _config = this.queue.some(p => this.isSameRequest(p, config));
        if (!_config) {
            let _this = this;
            config.cancelToken = new axios.CancelToken((cancel) => {
                this.queue.push({
                    url: config.url,
                    method: config.method,
                    cancel,
                    createTime: Date.now(),
                    dataMd5: _this.createDataMd5(config.data)  //根据实际情况是否支持data数据的计算md5
                })
            })
        } else {
            config.cancelToken = new axios.CancelToken((cancel) => {
                cancel('Duplicate request');
            })
        }
    },
    //删除超时请求
    removeRequestOverTime() {
        if (!this.canUsed()) return;
        const nowDate = Date.now();
        for (const q in this.queue) {
            const {createTime} = this.queue[q];
            const time = nowDate - createTime;
            if (time >= (this.limitTime)) {
                this.queue.splice(Number(q), 1);
            }
        }
    },
    //删除请求
    removeRequest(config) {
        if (!config) return;
        if (!this.canUsed(config)) return;
        for (const [index, p] of Object.entries(this.queue)) {
            if (this.isSameRequest(p, config)) {
                if (p.createTime) {
                    if ((Date.now() - p.createTime) > (this.limitTime)) {
                        this.queue.splice(Number(index), 1);
                    } else {
                        if (p.cancel) p.cancel('Duplicate request');
                    }
                } else {
                    this.queue.splice(Number(index), 1);
                }
                break;
            }

        }
    },

};


axios.defaults.headers["Content-Type"] = "application/json;charset=utf-8"
// 创建axios实例
const service = axios.create({
    // axios中请求配置有baseURL选项,表示请求URL公共部分
    baseURL: process.env.VUE_APP_BASE_API,
    // 超时
    timeout: 120000
})

// request拦截器
service.interceptors.request.use(
    (config) => {
        // 是否需要设置 token
        const isToken = (config.headers || {}).isToken === false
        if (getToken() && !isToken) {
            config.headers["Authorization"] = "Bearer " + getToken() // 让每个请求携带自定义token 请根据实际情况自行修改
        }
        // get请求映射params参数
        if (config.method === "get" && config.params) {
            let url = config.url + "?" + tansParams(config.params)
            url = url.slice(0, -1)
            config.params = {}
            config.url = url
        }
        //2023-11-03 请求重复限制
        if (config.url) {
            requestQueue.removeRequest(config)
            requestQueue.addRequest(config);
        }
        return config
    },
    (error) => {
        console.log(error)
        Promise.reject(error)
    }
)

// 响应拦截器
service.interceptors.response.use(
    (res) => {
        //2023-11-03 请求重复限制
        requestQueue.removeRequest(res.config)
        // 未设置状态码则默认成功状态
        const code = res.data.code || 200
        // 获取错误信息
        const msg = errorCode[code] || res.data.msg || errorCode["default"]
        // 二进制数据则直接返回
        if (
            res.request.responseType === "blob" ||
            res.request.responseType === "arraybuffer"
        ) {
            return res.data
        }
        if (code === 401) {
            if (!isReloginShow) {
                isReloginShow = true
                MessageBox.confirm('登录状态已过期,您可以继续留在该页面,或者重新登录', '系统提示', {
                        confirmButtonText: "重新登录",
                        cancelButtonText: "取消",
                        type: "warning"
                    }
                ).then(() => {
                    isReloginShow = false
                    store.dispatch('LogOut').then(() => {
                        // 如果是登录页面不需要重新加载
                        if (window.location.hash.indexOf("#/login") != 0) {
                            location.href = getAbsolutePath('/index')
                        }
                    })
                }).catch(() => {
                    isReloginShow = false
                })
            }
            return Promise.reject("无效的会话,或者会话已过期,请重新登录。")
        } else if (code === 500) {
            Message({
                message: msg,
                type: "error"
            })
            return Promise.reject(new Error(msg))
        } else if (code !== 200) {
            Notification.error({
                title: msg
            })
            return Promise.reject("error")
        } else {
            return res.data
        }
    },
    (error) => {
        console.log("err" + error)
        let {message} = error
        if (message == "Network Error") {
            //message = "后端接口连接异常"
            message = "网络异常"
        } else if (message.includes("timeout")) {
            //message = "系统接口请求超时"
            requestQueue.removeRequestOverTime();
            message = "请求超时"
        } else if (message.includes("Request failed with status code")) {
            let t_port = message.substr(message.length - 3);
            if (t_port == '404') {
                message = "未找到系统资源"
            } else
                message = "系统接口" + message.substr(message.length - 3) + "异常"
        } else if (message.includes('Duplicate request')) {
            //2023-11-03
            message = "重复提交,请求太频繁"
        }

        Message({
            message: message,
            type: "error",
            duration: 5 * 1000
        })
        return Promise.reject(error)
    }
)


// 通用下载方法
export function download(url, params, filename) {
    downloadLoadingInstance = Loading.service({
        text: "正在下载数据,请稍候",
        spinner: "el-icon-loading",
        background: "rgba(0, 0, 0, 0.7)"
    })
    return service
        .post(url, params, {
            transformRequest: [
                (params) => {
                    return tansParams(params)
                }
            ],
            headers: {"Content-Type": "application/x-www-form-urlencoded"},
            responseType: "blob"
        })
        .then(async (data) => {
            const isLogin = await blobValidate(data)
            if (isLogin) {
                const blob = new Blob([data])
                saveAs(blob, filename)
            } else {
                const resText = await data.text()
                const rspObj = JSON.parse(resText)
                const errMsg = errorCode[rspObj.code] || rspObj.msg || errorCode['default']
                Message.error(errMsg)
            }
            downloadLoadingInstance.close()
        })
        .catch((r) => {
            console.error(r)
            Message.error("下载文件出现错误!")
            downloadLoadingInstance.close()
        })
}


// 通用下载方法 get
export function downloadGet(url, params, filename, callback) {
    downloadLoadingInstance = Loading.service({
        text: "正在下载数据,请稍候",
        spinner: "el-icon-loading",
        background: "rgba(0, 0, 0, 0.7)",
    })

    return service.get(url, {
        // transformRequest: [(params) => {
        //   return tansParams(params)
        // }],
        params: params,
        headers: {'Content-Type': 'application/x-www-form-urlencoded'},
        responseType: 'blob'
    }).then((data) => {
        downloadLoadingInstance.close();
        //console.log('service.get',data);
        const isLogin = blobValidate(data);
        if (isLogin) {
            const blob = new Blob([data]);
            if (callback) {
                callback(blob);
                return;
            }
            saveAs(blob, filename)
        } else {
            const resText = data.text();
            const rspObj = JSON.parse(resText);
            const errMsg = errorCode[rspObj.code] || rspObj.msg || errorCode['default']
            Message.error(errMsg);
        }
    }).catch((r) => {
        console.error(r)
        Message.error('下载文件出现错误,请联系管理员!')
        downloadLoadingInstance.close();
    })
}


export default service

2、 业务处理代码采取单例模式
3、 插入数据时使用SQL验重。

标签:return,数据,解决方案,data,提交,error,const,message,config
From: https://www.cnblogs.com/javacoffeenet/p/17811813.html

相关文章

  • std::sort 传入成员函数指针报错的解决方案
    问题引入有一个类A,A的某个成员函数需要对A的某些变量进行std::sort,同时要调用A的另一个成员函数作为比较器。如代码所示:structA{vector<int>pos={0,4,2,5,3};boolcmp(intx,inty){returnpos[x]<pos[y];}voiddemo(){vector<int>a={2......
  • 多种模态数据集
    图像描述ImageCaptioningLAION-5B2022.3发布的迄今为止最大规模的图文对的多模态数据集。共计约5.85B数据,是基于CLIP过滤的。基于这个大型数据集,作者也发布不同侧重的子集。LAION2B-en是包含英文注释文本的,LAION2B-multi是包含100多种的其它注释文本语言的,LAION2B-nolang其中......
  • 【专题】中国服务机器人产业研究报告PDF合集分享(附原数据表)
    原文链接:https://tecdat.cn/?p=34144原文出处:拓端数据部落公众号仿生机器人作为一类结合了仿生学原理的机器人,具备自主决策和规划行动的能力,正逐渐进入大众视野。它们的核心技术要素包括感知与认知技术、运动与控制技术、人机交互技术和自主决策技术。阅读原文,获取专题报告合集......
  • 【专题】2023年中国手术机器人行业专题报告PDF合集分享(附原数据表)
    原文链接:https://tecdat.cn/?p=34144原文出处:拓端数据部落公众号仿生机器人作为一类结合了仿生学原理的机器人,具备自主决策和规划行动的能力,正逐渐进入大众视野。它们的核心技术要素包括感知与认知技术、运动与控制技术、人机交互技术和自主决策技术。阅读原文,获取专题报告合集......
  • 喜讯!极限科技成功签约中国一汽搜索数据库三年许可订阅合同!
    中标喜讯!极限科技INFINI Easysearch 成功签约中国第一汽车股份有限公司三年订阅合同!一汽集团作为国内汽车行业龙头企业,数字化转型伴随业务发展不断深化,非结构化数据日益成为各类组织数据的增长主力,逐渐成为数据要素的重要组成部分。以自动分词技术、倒排索引技术、相关度计算、......
  • java 基本数据类型和引用数据类型02
    ......
  • 【MySQL】MVCC机制、ReadView数据结构、匹配规则详解
    (目录)MySQLMVCC机制1.隔离级别在MySQLInnoDB存储引擎下,RC、RR基于MVCC(多版本并发控制)进行并发事务控制MVCC是**基于”数据版本”**对并发事务进行访问2.场景分析UNDO_LOG不是会被删除吗?中间数据万一被删了版本链不就断了?UNDO_LOG版本链不是立即删除,MySQL确保版......
  • 数据结构的初认识
    一般,我们将数据结构分为逻辑结构和物理结构。逻辑结构:是指数据对象中数据元素的相互关系。逻辑结构包括:集合结构,线性结构,树型结构,图形结构。       物理结构:是指数据的逻辑结构在计算机中的存储形式。根据物理结构的定义,我们实际上研究的的就是如......
  • 程序员为啥要做副业(05)-业务解决方案缔造
    除了技术,副业也可以帮助我们在业务上获得新认知,保持敏感性。之前我们在做程序员职业成长服务的时候,发现了一个问题。很多初阶的程序员没法升到中高阶,有两个很大的非技术影响因素:1管理能力每个程序员即使把自己的潜力发挥到极致,成为十倍开发者(10xdeveloper),他可以处理的事情也......
  • 异常检测算法-完全卷积数据描述子FCDD
    文献来源:EXPLAINABLEDEEPONE-CLASSCLASSIFICATION   最近在做一些异物检测之类的算法任务,原本想使用目标识别算法,但是问题是正样本太多,而负样本没几个。所以有必要使用异常检测算法,日后不妨再结合目标识别任务去做。在正式开始前,需要先简单介绍一个广义损失函数的东西......