文件上傳的幾種實(shí)現(xiàn)方式
form-data、base64、縮略圖、進(jìn)度條、拖拽上傳、切片上傳
1. Form-Data方式上傳
主要使用form表單方式實(shí)現(xiàn)文件上傳
let formData = new FormData();
console.log(this.file);
formData.append('file', this.file);
formData.append('filename', this.file.name);
instance.post('/upload_single', formData).then(res => {
if (+res.code === 0) {
this.fileTip = '圖片上傳成功!'
return;
}
return Promise.reject(res.codeText);
}).catch(err => {
console.log(err);
})
2. BASE64方式上傳
使用FildReader獲取文件的base64,將其上傳
let base64 = await changeBase64(that.file);
try {
const data = await instance
.post('/upload_single_base64', {
file: encodeURIComponent(base64), // 防止亂碼
filename: that.file.name
},{
headers: {
'Content-Type': 'application/x-www-form-urlencoded',
}
})
const { code } = data;
if (code === 0) {
this.fileTip = '文件上傳成功!';
}
throw data.codeText; // 拋出異常
} catch (error) {
console.log(error);
}
3. 文件縮略圖顯示,文件hash獲取
根據(jù)文件內(nèi)容展示縮略圖(主要用于圖片),根據(jù)內(nèi)容獲取hash值判斷后端是否存在該文件,節(jié)省上傳時(shí)間
// 獲取文件hash值
const changeBuffer = (file) => {
return new Promise((resolve) => {
let fileReader = new FileReader();
fileReader.readAsArrayBuffer(file);
fileReader.onload = (e) => {
let buffer = e.target.result;
const spark = new SparkMD5.ArrayBuffer();
spark.append(buffer);
const HASH = spark.end();
const suffix = /\.([0-9a-zA-Z]+)$/.exec(file.name)[1];
resolve({
buffer,
HASH,
suffix,
filename: `${HASH}.${suffix}`,
});
};
});
};
// 展示縮略圖
let base64 = await changeBase64(file);
upload_abber_img.src = base64;
4. 文件上傳進(jìn)度條實(shí)現(xiàn)
展示文件上傳進(jìn)度
try {
let formData = new FormData();
formData.append('file', this.file);
formData.append('filename', this.file.name);
const data = await instance.post('/upload_single', formData, {
onUploadProgress: (e) => {
const { loaded, total } = e;
upload_progress.style.display = 'block';
upload_progrees_value.style.width = `${ (loaded / total) * 100 }%`;
},
});
if (+data.code === 0) {
upload_progrees_value.style.width = `100%`;
this.fileTip = '文件上傳成功!'
return;
}
throw data.codeText;
} catch (error) {
console.log(e);
this.fileTip = '文件上傳失敗!'
} finally {
upload_progress.style.display = 'none';
upload_progrees_value.style.width = `0%`;
}
5. 文件拖拽上傳
使用原生JS事件 dragenter
、drop
、dragover
實(shí)現(xiàn)文章來源:http://www.zghlxwxcb.cn/news/detail-598951.html
dragenter(e) {
e.preventDefault();
},
drop(e) {
e.preventDefault();
const {
dataTransfer: { files },
} = e;
const file = files[0];
this.fileTip = uploadFile(file);
},
dragover(e) {
e.preventDefault();
},
uploadChange(e) {
const file = e.target.files[0];
this.fileTip = uploadFile(file);
},
const uploadFile = (file) => {
const that = this;
if (!file) {
alert('請選擇您要上傳的文件~');
return;
}
let formData = new FormData();
formData.append('file', file);
formData.append('filename', file.name);
let flag = false;
flag = instance
.post('/upload_single', formData)
.then(res => {
if (res.code === 0) {
return true;
}
return Promise.reject(res.codeText);
})
.catch(err => {
console.log(err);
})
return flag ? '文件上傳成功!': '文件上傳失敗!';
}
6. 大文件切片上傳
- 首先確定上傳規(guī)模數(shù)量,判斷當(dāng)前文件可以切出多少切片,
- 利用Bole.prototype.slice() 方法對(duì)文件進(jìn)行切片并且對(duì)每個(gè)文件片作唯一標(biāo)識(shí)(HASH),
- 從服務(wù)器獲取已經(jīng)上傳的切片,判斷切片是否存在
- 將切片全部上傳后,后端進(jìn)行合并處理
let chunkList = [],
alreadyChunkList = [],
maxSize = 1024 * 1024,
maxCount = Math.ceil(this.file.size / maxSize),
index = 0;
const { HASH, suffix } = await this.fileHash(this.file);
if (maxCount > 10) {
// 如果切片數(shù)量大于最大值
maxSize =this.file.size / 10; // 則改變切片大小
maxCount = 10;
}
while (index < maxCount) {
chunkList.push({
file: this.file.slice(index * maxSize, (index + 1) * maxSize),
filename: `${HASH}_${index + 1}.${suffix}`,
});
index++;
}
// 先獲取已經(jīng)上傳的切片
const data = await instance.post(
'/upload_already',
{
HASH: HASH,
},
{
headers: {
'Content-Type': 'application/x-www-form-urlencoded',
},
}
);
index = 0;
const clear = () => {
upload_progress.style.display = 'none';
upload_progrees_value.style.width = '0%'
}
const complate = async () => {
index++;
upload_progress.style.display = 'block';
upload_progrees_value.style.width = `${ (index / maxCount) * 100 }%`;
if (index < maxCount) {
return;
}
upload_progrees_value.style.width = `100%`;
try {
let res = await instance
.post('/upload_merge',
{
HASH,
maxCount
},
{
headers: {'Content-Type': 'application/x-www-form-urlencoded'}
}
);
if (res.code === 0) {
this.fileTip = '大文件切片上傳成功!';
clear();
return;
}
throw data.codeText;
} catch (error) {
clear();
}
}
const { fileList } = data;
alreadyChunkList = fileList;
chunkList.forEach(item => {
if (alreadyChunkList.length > 0 && alreadyChunkList.includes(item.filename)) {
debugger
// 切片已經(jīng)存在
complate()
return;
}
const fm = new FormData();
fm.append('file', item.file);
fm.append('filename', item.filename);
instance
.post('/upload_chunk', fm)
.then(data => {
if (+data.code === 0) {
complate();
return;
}
return Promise.reject(data.codeText);
})
.catch(() => {
this.fileTip = '上傳失敗';
clear();
})
});
源碼 https://github.com/archer0621/fileUpload文章來源地址http://www.zghlxwxcb.cn/news/detail-598951.html
到了這里,關(guān)于基于Vue實(shí)現(xiàn)文件上傳的文章就介紹完了。如果您還想了解更多內(nèi)容,請?jiān)谟疑辖撬阉鱐OY模板網(wǎng)以前的文章或繼續(xù)瀏覽下面的相關(guān)文章,希望大家以后多多支持TOY模板網(wǎng)!