前端大文件上传
一、切片上传技术原理
切片上传是把大文件分割成多个较小的切片,分别上传这些切片,最后在服务器端将它们合并成完整文件。这种方式能有效应对网络不稳定导致的上传失败问题,还可利用多线程并行上传,提升上传效率。
二、前端实现步骤及与后端接口交互
1 文件选择与信息获取
用户通过文件选择框选择要上传的大文件。前端获取文件的基本信息,包括文件名、文件大小、文件类型等。这些信息将在后续的上传过程中发挥重要作用,例如文件名用于服务器端存储文件,文件大小用于计算切片数量和监控上传进度。
2 切片参数设定
根据服务器的性能和网络状况,合理确定切片的大小。一般来说,切片大小可以设置为 1MB - 5MB 之间。较小的切片在网络不稳定时能更好地实现断点续传,但会增加服务器合并的工作量;较大的切片则反之。
3 文件切片操作
使用 JavaScript 的 File.prototype.slice
方法对文件进行切片。从文件的起始位置开始,按照设定的切片大小依次截取文件片段,直到文件末尾。每个切片都有其在原文件中的偏移量和唯一标识,方便服务器识别和排序。
4 切片信息生成
为每个切片生成唯一的标识,可采用 UUID 算法。同时记录切片的索引、总切片数量、文件的唯一标识等信息。这些信息将与切片数据一起发送到服务器,用于服务器端的切片合并和文件完整性验证。
5 预上传请求(与后端接口 1:预检查接口)
接口功能
前端向后端发送预上传请求,携带文件的基本信息(文件名、文件大小、文件哈希值等)。后端接收到请求后,检查该文件是否已存在于服务器中。若存在,则返回已上传的切片信息,前端可以根据这些信息跳过已上传的切片,直接上传未完成的部分;若不存在,则为此次上传创建记录,并返回允许上传的标识,前端可以开始上传所有切片。
接口请求方式
通常采用 POST 请求,将文件信息以 JSON 格式或表单数据的形式发送到后端。
接口返回数据
-
若文件已存在:返回已上传的切片索引列表。
-
若文件不存在:返回一个唯一的上传标识,用于后续切片上传和合并操作的关联。
6 切片上传请求(与后端接口 2:切片上传接口)
接口功能
前端根据预上传请求的结果,将未上传的切片逐个发送到后端。每个请求携带切片数据、切片信息(索引、总切片数量、文件唯一标识等)。后端接收到切片后,将其存储到临时目录,并记录切片的存储路径和相关信息。
接口请求方式
采用 POST 请求,请求体为 multipart/form-data
格式,包含切片文件和切片信息。
接口返回数据
返回上传成功的标识或状态码,前端根据返回结果更新上传进度和状态。
3.7 上传进度监控
在切片上传过程中,前端实时监控上传进度。可以通过监听请求的 progress
事件,获取已上传的字节数和总字节数,计算出上传进度百分比,并将进度信息展示给用户。同时,将上传进度信息存储在本地,以便在上传中断后恢复时使用。
3.8 断点续传处理
若上传过程中出现网络中断或其他异常情况,前端记录已上传的切片信息。重新上传时,再次调用预上传接口,获取未上传的切片列表,继续上传未完成的切片,实现断点续传功能。
9 合并请求(与后端接口 3:切片合并接口)
接口功能
当所有切片上传完成后,前端向后端发送合并请求,携带文件唯一标识和总切片数量。后端接收到请求后,根据记录的切片信息,按顺序将切片合并成完整文件,并进行完整性验证。验证通过后,将合并后的文件移动到指定存储位置,并清理临时切片文件。
接口请求方式
采用 POST 请求,将文件唯一标识和总切片数量以 JSON 格式或表单数据的形式发送到后端。
接口返回数据
返回合并成功的标识或状态码,以及合并后文件的存储路径或访问链接。
<template><div class="break-point"><div class="gva-table-box"><el-divider content-position="left">大文件上传</el-divider><form id="fromCont" method="post"><div class="fileUpload" @click="inputChange">选择文件<input v-show="false" id="file" ref="FileInput" multiple="multiple" type="file" @change="choseFile"></div></form><el-button :disabled="limitFileSize" type="primary" size="small" class="uploadBtn" @click="getFile">上传文件</el-button><div class="el-upload__tip">请上传不超过5MB的文件</div><div class="list"><transition name="list" tag="p"><div v-if="file" class="list-item"><el-icon><document /></el-icon><span>{{ file.name }}</span><span class="percentage">{{ percentage }}%</span><el-progress :show-text="false" :text-inside="false" :stroke-width="2" :percentage="percentage" /></div></transition></div><div class="tips">此版本为先行体验功能测试版,样式美化和性能优化正在进行中,上传切片文件和合成的完整文件分别再QMPlusserver目录的breakpointDir文件夹和fileDir文件夹</div></div></div></template><script setup>
import SparkMD5 from 'spark-md5'
import {findFile,breakpointContinueFinish,removeChunk,breakpointContinue
} from '@/api/breakpoint'
import { ref, watch } from 'vue'
import { ElMessage } from 'element-plus'
import { ElLoading } from 'element-plus'const file = ref(null) // 读取的文件信息,可以读取文件大小、同时生成MD5标识等等。
const fileMd5 = ref('') // Md5加密标识
const formDataList = ref([]) // 分片存储的一个池子,{ key: sliceIndex.value, formData }
const waitUpLoad = ref([]) // 当是断点续传,剩下没有上传的切片
const waitNum = ref(NaN) // 当是断点续传,剩下没有上传的切片数量
const limitFileSize = ref(false)
const percentage = ref(0)
const percentageFlage = ref(true)
const FileSliceCap = 10 * 1024 * 1024 // 分片字节数,如果分的太细需要发送很多个分片请求。
const sliceStart = ref(0) // 定义分片开始切的地方
const sliceEnd = ref(0) // 每片结束切的地方a
const sliceIndex = ref(0) // 第几片
const allChunkSize = ref(0) // 所有大切片的累加体积
const isCreateMd5 = ref(true) // 是否生成Md5
const limitSize = 2000 * 1024 * 1024 // 浏览器对于上传的文件有限定大小,大于2G的文件无法上传。我们需要对大于2G的文件先切成大的chunk
const ChunkSizeCap = 2000 * 1024 * 1024 // 大的chunk容量
const maxSize = 10000 * 1024 * 1024 // 限制上传的文件大小
let moreThan_2G = falseconst init = () => {percentage.value = 0allChunkSize.value = 0sliceStart.value = 0sliceEnd.value = 0sliceIndex.value = 0moreThan_2G = file.value.size > limitSizeisCreateMd5.value = true
}// 选中文件的函数。如果文件大于2G,先进行一次切片。
const choseFile = async(e) => {const fileInput = e.target.files[0] // 获取当前文件file.value = fileInputinit()if (file.value.size < maxSize) {if (moreThan_2G) {formDataList.value = []let chunkStart = 0let chunkEnd = 0let chunkIndex = 0while (chunkEnd < file.value.size) {chunkStart = chunkIndex * ChunkSizeCap // 计算每片开始位置chunkEnd = (chunkIndex + 1) * ChunkSizeCap // 计算每片结束位置const fileChunk = file.value.slice(chunkStart, chunkEnd)sliceChunk(fileChunk)chunkIndex++}} else {sliceChunk(file.value)}} else {limitFileSize.value = trueElMessage('请上传小于5M文件')}
}/*** 对文件进行切片,同时根据文件信息生成md5,md5用于后端识别文件。* 切片信息包括:* -md5* -当前切片文件* -当前是第几片* -文件名* @param fileChunk*/
const sliceChunk = (fileChunk) => {const fileR = new FileReader() // 创建一个reader用来读取文件流percentage.value = 0fileR.readAsArrayBuffer(fileChunk) // 把文件读成ArrayBuffer 主要为了保持跟后端的流一致const loading = ElLoading.service({lock: true,text: 'Loading',background: 'rgba(0, 0, 0, 0.7)',})fileR.onload = async e => {allChunkSize.value += fileChunk.sizeif (isCreateMd5.value) {// 读成arrayBuffer的回调 e 为方法自带参数 相当于 dom的e 流存在e.target.result 中const blob = e.target.resultconst spark = new SparkMD5.ArrayBuffer() // 创建md5制造工具spark.append(blob) // 文件流丢进工具fileMd5.value = spark.end() // 工具结束 产生一个总文件的md5isCreateMd5.value = false}// 当结尾数字大于文件总size的时候 结束切片while (sliceEnd.value < file.value.size) {sliceStart.value = sliceIndex.value * FileSliceCap // 计算每片开始位置sliceEnd.value = (sliceIndex.value + 1) * FileSliceCap // 计算每片结束位置var fileSlice = fileChunk.slice(sliceStart.value, sliceEnd.value) // 开始切 file.slice 为 h5方法 对文件切片 参数为 起止字节数const formData = new window.FormData() // 创建FormData用于存储传给后端的信息formData.append('fileMd5', fileMd5.value) // 存储总文件的Md5 让后端知道自己是谁的切片formData.append('file', fileSlice) // 当前的切片formData.append('chunkNumber', sliceIndex.value) // 当前是第几片formData.append('fileName', file.value.name) // 当前文件的文件名 用于后端文件切片的命名 formData.appen 为 formData对象添加参数的方法formDataList.value.push({ key: sliceIndex.value, formData }) // 把当前切片信息 自己是第几片 存入我们方才准备好的池子sliceIndex.value++}if (allChunkSize.value === file.value.size) {loading.close()submitSlice()}}fileR.onerror = async e => {loading.close()ElMessage('文件读取失败')}
}/*** 提交给后端要上传的文件信息,包括md5,切片数量等等。* 后端根据前端生成的md5与数据库文件的md5进行对比,如果一致那么就表示上传完成实现了秒传功能。* 如果数据库没有上传成功的文件,那么后端会返回对应文件已经上传的切片数据。* 前端formDataList和对应文件已经上传的切片数据进行对比,判断还需要上传哪些切片,从而实现断点上传。* @returns {Promise<void>}*/
const submitSlice = async() => {const params = {fileName: file.value.name,fileMd5: fileMd5.value,chunkTotal: formDataList.value.length}console.log(params)const res = await findFile(params)// 全部切完以后 发一个请求给后端 拉当前文件后台存储的切片信息 用于检测有多少上传成功的切片const finishList = res.data.file.ExaFileChunk // 上传成功的切片const IsFinish = res.data.file.IsFinish // 是否是同文件不同命 (文件md5相同 文件名不同 则默认是同一个文件但是不同文件名 此时后台数据库只需要拷贝一下数据库文件即可 不需要上传文件 即秒传功能)if (!IsFinish) {// 当是断点续传时候waitUpLoad.value = formDataList.value.filter(all => {return !(finishList &&finishList.some(fi => fi.FileChunkNumber === all.key))})} else {waitUpLoad.value = [] // 秒传则没有需要上传的切片ElMessage.success('文件已秒传')}waitNum.value = waitUpLoad.value.length // 记录长度用于百分比展示
}const getFile = () => {// 确定按钮if (file.value === null) {ElMessage('请先上传文件')return}if (percentage.value === 100) {percentageFlage.value = false}sliceFile() // 上传切片
}/*** 获取当前切片md5,主要用于后端验证切片完整性*/
const sliceFile = () => {waitUpLoad.value &&waitUpLoad.value.forEach(item => {// 需要上传的切片item.formData.append('chunkTotal', formDataList.value.length) // 切片总数携带给后台 总有用的const fileR = new FileReader() // 功能同上const fileF = item.formData.get('file')fileR.readAsArrayBuffer(fileF)fileR.onload = e => {const spark = new SparkMD5.ArrayBuffer()spark.append(e.target.result)item.formData.append('chunkMd5', spark.end())upLoadFileSlice(item)}})
}watch(() => waitNum.value, () => {percentage.value = Math.floor(((formDataList.value.length - waitNum.value) / formDataList.value.length) * 100)
})/*** 上传切片信息,有多少个切片就发送多少个请求,因此切片大小和数量要根据实际情况去定。* 切片上传完毕以后,还要给后端再次发送一次请求,让后端合成文件并删除缓存切片。*/
const upLoadFileSlice = async(item) => {// 切片上传const fileRe = await breakpointContinue(item.formData)if (fileRe.code !== 0) {return}waitNum.value-- // 百分数增加if (waitNum.value === 0) {// 切片传完以后 合成文件const params = {fileName: file.value.name,fileMd5: fileMd5.value}const res = await breakpointContinueFinish(params)if (res.code === 0) {// 合成文件过后 删除缓存切片const params = {fileName: file.value.name,fileMd5: fileMd5.value,filePath: res.data.filePath,}ElMessage.success('上传成功')await removeChunk(params)}}
}const FileInput = ref(null)
const inputChange = () => {FileInput.value.dispatchEvent(new MouseEvent('click'))
}
</script><script>export default {name: 'BreakPoint'
}
</script><style lang="scss" scoped>
h3 {margin: 40px 0 0;
}ul {list-style-type: none;padding: 0;
}li {display: inline-block;margin: 0 10px;
}a {color: #42b983;
}#fromCont {display: inline-block;
}.fileUpload {padding: 3px 10px;font-size: 12px;height: 20px;line-height: 20px;position: relative;cursor: pointer;color: #000;border: 1px solid #c1c1c1;border-radius: 4px;overflow: hidden;display: inline-block;input {position: absolute;font-size: 100px;right: 0;top: 0;opacity: 0;cursor: pointer;}
}.fileName {display: inline-block;vertical-align: top;margin: 6px 15px 0 15px;
}.uploadBtn {position: relative;top: -10px;margin-left: 15px;
}.tips {margin-top: 30px;font-size: 14px;font-weight: 400;color: #606266;
}.el-divider {margin: 0 0 30px 0;
}.list {margin-top: 15px;
}.list-item {display: block;margin-right: 10px;color: #606266;line-height: 25px;margin-bottom: 5px;width: 40%;.percentage {float: right;}
}.list-enter-active, .list-leave-active {transition: all 1s;
}.list-enter, .list-leave-to/* .list-leave-active for below version 2.1.8 */
{opacity: 0;transform: translateY(-30px);
}
</style>
import request from '@/utils/request'// 查找文件
export const findFile = (params) => {return request({url: '/fileUploadAndDownload/findFile',method: 'post',data: params})
}// 断点续传
export const breakpointContinue = (data) => {return request({url: '/fileUploadAndDownload/breakpointContinue',method: 'post',data: data})
}// 断点续传完成
export const breakpointContinueFinish = (params) => {return request({url: '/fileUploadAndDownload/breakpointContinueFinish',method: 'post',data: params})
}// 删除切片
export const removeChunk = (params) => {return request({url: '/fileUploadAndDownload/removeChunk',method: 'post',data: params})
}
相关文章:
前端大文件上传
一、切片上传技术原理 切片上传是把大文件分割成多个较小的切片,分别上传这些切片,最后在服务器端将它们合并成完整文件。这种方式能有效应对网络不稳定导致的上传失败问题,还可利用多线程并行上传,提升上传效率。 二、前端实现…...
【网络】实现电脑与笔记本电脑之间的直接网络连接
要实现电脑与笔记本电脑之间的直接网络连接,可以通过有线或无线两种方式。以下是详细的步骤指南: 一、有线直连(通过网线) 1. 准备工具 网线:使用交叉网线(适用于旧设备)或普通直连网线&#…...
“深入浅出”系列之音视频开发:(12)使用FFmpeg实现倍速播放:技术细节与优化思路
一、前言 在音视频处理领域,倍速播放是一个常见的需求,尤其是在视频播放器、在线教育平台等场景中,用户常常需要以不同的速度播放视频内容。然而,实现一个高质量的倍速播放功能并不容易,尤其是在处理音频时࿰…...
qt作业day2
1:在注册登录的练习里面,追加一个QListWidget 项目列表 要求:点击注册之后,将账号显示到 listWidget上面去 以及,在listWidget中双击某个账号的时候,将该账号删除 .h #ifndef WIDGET_H #define WIDGET_H …...
Qt:day1
一、作业 写1个Widget窗口,窗口里面放1个按钮,按钮随便叫什么; 创建2个Widget对象: Widget w1, w2; w1.show(); w2不管; 要求: 点击 w1.btn,w1隐藏,w2显示; 点击 w2.btn&…...
基于微信小程序的停车场管理系统的设计与实现
第1章 绪论 1.1 课题背景 随着移动互联形式的不断发展,各行各业都在摸索移动互联对本行业的改变,不断的尝试开发出适合于本行业或者本公司的APP。但是这样一来用户的手机上就需要安装各种软件,但是APP作为一个只为某个公司服务的一个软件&a…...
详细Linux基础知识(不断完善)
终端类型分类 1. 物理终端 直接连接到计算机的硬件设备2. 虚拟终端 通过快捷键切换的文本模式界面: Ctrl + Alt + F1 # 登录窗口 Ctrl + Alt + F2 # 当前图形界面 Ctrl + Alt + F3 # 虚拟命令终端 Ctrl + Alt + F4-F6 # 备用虚拟终端3. 图形终端 模拟终端:图形环境中的…...
类和对象-继承-C++
1.定义 面向对象的三大特征之一,为了减少重复的代码 2.语法 class 子类 :继承方式 父类 (子类也叫派生类,父类也称为基类) 例:class age:public person; #include<iostrea…...
初阶数据结构(C语言实现)——3顺序表和链表(1)
目录 【本节目标】1. 线性表2.顺序表2.1概念及结构2.2 接口实现2.2.0 动态顺序表2.2.1 顺序表初始化SLInit()2.2.2 销毁和打印2.2.3 尾插SLPushBack()2.2.4 尾删SLPopBack()2.2.5 头插2.2.6 头删2.2.7 插入…...
nuxt常用组件库html-validator、@nuxtjs/i18n、@nuxt/image、@unocss/nuxt使用解析
html-validator 主要用于自动验证nuxt服务器呈现的HTML(SSR和SSG),以检测可能导致水合错误的HTML常见问题,有助于减少水合错误,检测常见的可访问性错误。 安装 npx nuxilatest module add html-validator配置 若自动更新nuxt.config.ts配置文…...
4G工业路由器在公交充电桩中的应用与优势
随着电动公交车的普及,公交充电桩的稳定运行和高效管理是交通营运部门最关心的问题。4G工业路由器凭借其卓越的数据采集和通讯能力,成为实现充电桩智能化管理的关键。 公交充电桩运维管理需求概述: 1.实时性:实时监控充电状态、剩…...
matlab 四维数据可视化(已解决)
虽然这不是传统意义上的“4维可视化”,但你可以通过在三维空间中表示两个维度来间接展示4维数据。例如,你可以使用颜色来表示第四个维度。 clc clear close all% 假设X, Y, Z为你的三维数据,C为第四维数据 X rand(100, 1); Y rand(100, 1);…...
歌曲分类和流行度预测
1. 项目介绍 本项目从kaggle平台上下载了数据集,该数据集包含了3万多首来自Spotify API 的歌曲,共有23个特征。首先对数据集进行预处理,如重复行、缺失值、标准化处理等。再对预处理后的数据进行探索性分析,观察各变量的分布情况&…...
经验分享:用一张表解决并发冲突!数据库事务锁的核心实现逻辑
背景 对于一些内部使用的管理系统来说,可能没有引入Redis,又想基于现有的基础设施处理并发问题,而数据库是每个应用都避不开的基础设施之一,因此分享个我曾经维护过的一个系统中,使用数据库表来实现事务锁的方式。 之…...
oracle decode
1. 基本语法 DECODE(expression, search1, result1, search2, result2, ..., default_result) expression :需要比较的表达式或列。search1, search2, ... :要匹配的值。result1, result2, ... :当 expression 等于 search 时返回的结果。def…...
让后台界面布局更灵活:在GrapesJS中复刻Java的五区式布局
当你想要在可视化编辑器中做一个类似Java BorderLayout 的五区布局,却发现市面上大多只能“简单拼接”而难以自由扩展时,你或许就需要一个更灵活的布局管理器来帮忙。本篇文章就从这个痛点开始,带你一步步揭秘如何用 GrapesJS 自定义并实现一…...
【网络安全 | 漏洞挖掘】分享21个基础漏洞案例
未经许可,不得转载。 文章目录 案例1:绕过500状态码案例2:修改前端实现任意文件上传案例3:Nmap扫描端口命令案例4:绕过限制实现任意文件读取案例5:删除任意目录文件案例6:锁定任意账户案例7:重置任意用户密码案例8:接管任意账户方法一方法二案例9:功能校验机制绕过案…...
期权适合什么类型的投资者交易?
财顺小编本文主要介绍期权适合什么类型的投资者交易?期权适合的投资者类型需结合其风险偏好、投资目标及市场判断能力综合评估。 期权适合什么类型的投资者交易? 1. 风险管理型投资者(稳健型) 适用场景:持有股票、大…...
浅谈C++/C命名冲突
前言 在这里我会简要地介绍产生命名冲突的原因,和C中处理命名冲突的方法,同时和C语言的解决办法进行比较。 相信你在阅读完之后一定会有收获。对于我们来说,了解编译器的编译链接过程才能更好的理解编译器是如何报错的,更能让我们…...
【星云 Orbit • STM32F4】08. 用判断数据头来接收据的串口通用程序框架
【星云 Orbit • STM32F4】08. 用判断数据头来接收据的串口通用程序框架 1. 引言 本教程旨在帮助嵌入式开发小白从零开始,学习如何在STM32F407微控制器上实现一个基于串口的数据接收程序。该程序能够通过判断数据头来接收一串数据,并将其存储到缓冲区中…...
C# OnnxRuntime部署DAMO-YOLO香烟检测
目录 说明 效果 模型信息 项目 代码 下载 参考 说明 效果 模型信息 Model Properties ------------------------- --------------------------------------------------------------- Inputs ------------------------- name:input tensor:Floa…...
探索Elasticsearch:索引的CRUD
在企业环境中,Elasticsearch的索引CRUD(创建Create、读取Read、更新Update、删除Delete)操作是非常基础且频繁使用的功能。这些操作对于管理和维护数据至关重要,尤其是在处理大规模数据集和需要实时搜索与分析的应用场景中。 目录…...
C++:多态与虚函数
1.虚函数,在函数前加virtual即可。有虚函数时,父类指针指向父类对象时就会使用父类的成员,指向子类对象时就可以使用子类成员,进而我们引入了多态的概念。 2.多态:父类指针指向子类的对象,通过父类指针调用…...
【Python 数据结构 2.时间复杂度和空间复杂度】
Life is a journey —— 25.2.28 一、引例:穷举法 1.单层循环 所谓穷举法,就是我们通常所说的枚举,就是把所有情况都遍历了的意思。 例:给定n(n ≤ 1000)个元素ai,求其中奇数有多少个 判断一…...
【Qt QML】QML鼠标事件(MouseArea)
QML鼠标事件全面解析 一、MouseArea基础概念 在 QML 中,鼠标事件是处理用户与界面元素交互的重要部分。QML 提供了多种方式来处理鼠标事件,MouseArea 是 QML 中用于处理鼠标事件的核心元素,它可以覆盖在其他元素之上,捕获鼠标操作并触发相应的信号。 1、基本用法 import …...
医脉云枢:中医药典籍知识图谱与非遗传承多维可视化系统
核心优势: "医脉"直击主题,"云枢"体现技术前瞻性 "非遗传承"呼应二十大文化政策 "多维"涵盖3D模型、时间轴、地图等多种可视化形式 技术栈:Vue Flask Element UI ECharts MySQL 同时参考了…...
vue 和 react 底层采用的 diff 算法的区别
Vue 3 和 React 在底层 Diff 算法上的实现确实有一些区别,主要体现在设计理念、性能优化策略以及具体实现方式上。以下是对两者 Diff 算法差异的详细分析: 1. 总体设计理念 Vue 3 的 Diff 算法 Vue 3 的虚拟 DOM Diff 算法基于“双端比较”思想ÿ…...
养老小程序方案详解居家养老小程序系统
养老小程序,上门居家养老小程序,用户端护工端小程序,管理后台。php开发语言,可源码搭建,二次开发或者定制开发。 一 用户端:小程序 核心功能模块:用户完善个人健康档案,在线选择服…...
Tauri跨平台开发问题及解决方案深度解析(React版)
Tauri跨平台开发问题及解决方案深度解析(React版) 一、环境配置与项目初始化难题(React适配) 1.1 React项目初始化 推荐模板: # 使用ReactTypeScript模板 npm create tauri-applatest -- --template react-ts# 项目…...
【OMCI实践】omci.lua脚本文件(独家分享)
引言 omci.lua文件是Wireshark的OMCI协议解析插件的核心组件。它配合BinDecHex.lua,可以解析OMCI协议的数据包,提取出消息类型、受管实体标识、受管实体属性等关键信息,并以人类可读的形式显示在Wireshark的解码视图中,方便研发人…...
React高级内容探索
flushSync确保了DOM立即更新 flushSync让你强制React同步刷新提供回调中的任何更新,这确保了DOM立即更新 flushSync是DOM更新之后的,像vue中的nextTick: import { useState,useRef} from "react" import { flushSync} from &quo…...
前端水印实现方式
一、简介 简单来说,前端水印就是在网页或应用程序的前端界面上添加的一种标记,通常是文本、图标或图案等形式。它就像给你的数字内容贴上了一个独特的 “标签”,用于标识内容的归属、防止未经授权的使用和传播。比如,一些在线图片…...
【新手入门】SQL注入之getshell(木马)
木马介绍 木马其实就是一段程序,这个程序运行到目标主机上时,主要可以对目标进行远程控制、盗取信息等功能,一般不会破坏目标主机,当然,这也看黑客是否想要搞破坏。 按照功能分类:远控型、破坏型、流氓软件型、盗取信…...
Git操作指南:分支合并、回退及其他重要操作
在软件开发的协作过程中,Git 作为一款强大的版本控制系统,能帮助开发者高效管理代码的各个版本和分支。本文将详细介绍 Git 中常见的分支合并、取消本地修改、回退操作等,并提供通俗易懂的解释和步骤指南。 一、分支合并 分支合并是 Git 工…...
Vue 3 引入全局状态管理 pinia
什么是全局状态管理? 所有页面全局共享的变量,而不是局限在某一个页面中。适合作为全局状态的数据: 已登录用户信息 (每个页面几乎都要用) Pinia 是一个主流的状态管理库,相比于 Vuex 来说使用更简单,可参考 入门文档 开始 | Pinia 进行引…...
SQL 中为什么参数多了not in 比 in 慢多了,怎么优化
开发工作中,我发现一个现象,比喻下面的两个语句: select * from shangpin where spdm in (1,2,3,...); select * from shangpin where spdm not in (1,2,3,...); 当参数比较少的时候还看不出来什么,但是遇到参数上了几百几千&am…...
[密码学实战]Java生成SM2根证书及用户证书
前言 在国密算法体系中,SM2是基于椭圆曲线密码(ECC)的非对称加密算法,广泛应用于数字证书、签名验签等场景。本文将结合代码实现,详细讲解如何通过Java生成SM2根证书及用户证书,并深入分析其核心原理。 一、证书验证 1.代码运行结果 2.根证书验证 3.用户证书验证 二、…...
API接口:企业名称、注册号、统一社会信用代码、企业类型、成立日期和法定代表人等数据 API 接口使用指南
API接口:企业名称、注册号、统一社会信用代码、企业类型、成立日期和法定代表人等数据 API 接口使用指南 本文详细介绍一种基于 Web 搜索方式实现的企业信息查询接口,适用于数据补全、企业资质验证、信息查询等场景。文章内容涵盖接口功能、请求参数、返…...
版图自动化连接算法开发 00004 ------ 给定一个点,添加一个中间点实现 Manhattan 方式连接两个给定的坐标点
版图自动化连接算法开发 00004 ------ 给定一个点,添加一个中间点实现 Manhattan 方式连接两个给定的坐标点 引言正文引言 必读文章 ------ 版图自动化连接算法开发 00001 ------ 直接连接两个给定的坐标点。 此处,我们对给定点的坐标进行一下限制,因为是只添加一个点,因…...
医疗行业电脑终端如何防病毒——火绒企业版杀毒软件
医疗物联网技术广泛应用,使得医院网络空间中增加了诸多新型终端设备。这些设备类型多样、型号各异,风险暴露面积大。火绒安全对医疗机构终端安全出现的问题、不足、需求等,提出整体解决方案。 医疗行业终端安全防护痛点 系统老旧 医院、区…...
【入门Web安全之前端学习的侧重点和针对性的建议】
入门Web安全之前端学习的侧重点和针对性的建议 一、HTML:理解攻击载荷的载体二、CSS:次要但需警惕点击劫持三、JavaScript:渗透测试的核心重点四、浏览器工具:渗透测试的实战武器五、学习建议与资源六、总结:渗透测试者…...
2024年中国城市统计年鉴(PDF+excel)
2024年中国城市统计年鉴(PDFexcel) 说明:包括地级县级市 格式:PDFEXCEL 《中国城市统计年鉴》是一部全面反映中国城市发展状况的官方统计出版物,包括各级城市的详细统计数据。这部年鉴自1985年开始出版,…...
详解DeepSeek模型底层原理及和ChatGPT区别点
一、DeepSeek大模型原理 架构基础 DeepSeek基于Transformer架构,Transformer架构主要由编码器和解码器组成,在自然语言处理任务中,通常使用的是Transformer的解码器部分。它的核心是自注意力机制(Self - Attention),这个机制允许模型在处理输入序列时,关注序列中不同位…...
Github 2025-03-03 开源项目周报Top14
根据Github Trendings的统计,本周(2025-03-03统计)共有14个项目上榜。根据开发语言中项目的数量,汇总情况如下: 开发语言项目数量Python项目5TypeScript项目4Jupyter Notebook项目3Go项目2JavaScript项目2C++项目2Vue项目1Rust项目1Dify.AI: 开源的LLM应用程序开发平台 创建…...
探秘基带算法:从原理到5G时代的通信变革【二】Viterbi解码
文章目录 二、关键算法原理剖析2.1 Viterbi 解码2.1.1 卷积码与网格图基础**卷积码****网格图****生成多项式****理想情况下解码过程** 2.1.2 Viterbi 算法核心思想2.1.3 路径度量与状态转移机制2.1.4 算法流程与关键步骤详解2.1.5 译码算法举例与复杂度分析2.1.6 算法代码示例…...
MAC 本地搭建部署 dify(含 github访问超时+Docker镜像源拉取超时解决方案)
目录 一、什么是 dify? 二、安装 docker 1. 什么是 docker? 2. docker下载地址 三、安装 dify 1. dify下载地址 2.可能遇到问题一: github访问超时 3.下载后完成解压 4.进入到 cmd 终端环境,执行下面三个命令 5.可能遇到…...
扫描纸质文件转pdf---少页数+手机+电脑协作
针对手机上扫描软件扫描文件转pdf要收费的问题,提供一种在页数较少时的免费替代方案 。 实现方法:手机软件的免费功能将文件扫描并保存为图片电脑端在word中将图片拼成文档word转pdf 1.借助于“扫描全能王”APP可以免费扫描文件为图片的功能࿰…...
[KEIL]单片机技巧 01
1、查看外设寄存器的值 配合对应的芯片开发手册以查看寄存器及其每一位的意义,可以解决90%以上的单纯的片内外设bug,学会如何通过寄存器的值来排外设上的蛊是嵌入式开发从小白到入门的重要一步,一定要善于使用这个工具,而不是外设…...
B站上优质的Java和SpringBoot相关视频教程
Java和SpringBoot相关视频教程 Java大师课 【Java】上部:Java Master Class大师课教程【Java】下部:Java Master Class大师课教程 Spring Boot & React 【Spring Boot & React】Spring Boot和React教程 Spring Boot 【Spring Boot】Spring…...
Spring学习笔记04:spring mvc和Spring Boot之间是什么关系?
Spring MVC 是什么? 想象你开了一家餐厅,顾客(用户)点菜、服务员传话、厨师做菜、最后服务员上菜。Spring MVC 就是规定这套流程的“餐厅管理规则”,专门用于处理网页请求(HTTP)和响应。 核心…...