基于Vue实现文件上传的几种实现方式

 更新时间:2024年03月27日 09:24:50   作者:freshfish丶  
文件上传是web开发中一个常见的需求,Vue.js作为一款流行的前端框架,也提供了方便的方法来实现文件上传功能,下面这篇文章主要给大家介绍了关于基于Vue实现文件上传的几种实现方式,需要的朋友可以参考下
(福利推荐:【腾讯云】服务器最新限时优惠活动,云服务器1核2G仅99元/年、2核4G仅768元/3年,立即抢购>>>:9i0i.cn/qcloud

(福利推荐:你还在原价购买阿里云服务器?现在阿里云0.8折限时抢购活动来啦!4核8G企业云服务器仅2998元/3年,立即抢购>>>:9i0i.cn/aliyun

文件上传的几种实现方式

form-data、base64、缩略图、进度条、拖拽上传、切片上传

1. Form-Data方式上传

主要使用form表单方式实现文件上传

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获取

根据文件内容展示缩略图(主要用于图片),根据内容获取hash值判断后端是否存在该文件,节省上传时间

// 获取文件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. 文件上传进度条实现

展示文件上传进度

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 实现

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. 大文件切片上传

  • 首先确定上传规模数量,判断当前文件可以切出多少切片,
  • 利用Bole.prototype.slice() 方法对文件进行切片并且对每个文件片作唯一标识(HASH),
  • 从服务器获取已经上传的切片,判断切片是否存在
  • 将切片全部上传后,后端进行合并处理
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) {
    // 如果切片数量大于最大值
    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++;
}

// 先获取已经上传的切片
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
        // 切片已经存在
        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

总结

到此这篇关于基于Vue实现文件上传的几种实现方式的文章就介绍到这了,更多相关Vue实现文件上传内容请搜索程序员之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持程序员之家!

相关文章

  • vue3动态添加路由

    vue3动态添加路由

    这篇文章主要介绍了vue3动态添加路由,文章围绕主题展开详细的内容介绍,具有一定的参考价值,需要的小伙伴可以参考一下
    2022-06-06
  • vue项目中icon乱码的问题及解决

    vue项目中icon乱码的问题及解决

    这篇文章主要介绍了vue项目中icon乱码的问题及解决方案,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2022-12-12
  • vue-element-admin如何设置默认语言

    vue-element-admin如何设置默认语言

    这篇文章主要介绍了vue-element-admin如何设置默认语言,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2022-04-04
  • vue iview实现动态路由和权限验证功能

    vue iview实现动态路由和权限验证功能

    这篇文章主要介绍了vue iview实现动态路由和权限验证功能,动态路由控制分为两种:一种是将所有路由数据存储在本地文件中,另一种则是本地只存储基本路由,具体内容详情大家参考下此文
    2018-04-04
  • vue?3.0使用element-plus按需导入方法以及报错解决

    vue?3.0使用element-plus按需导入方法以及报错解决

    Vue3是不能直接使用Element-ui了,需要换成Element-plus,下面这篇文章主要给大家介绍了关于vue?3.0使用element-plus按需导入方法以及报错解决的相关资料,需要的朋友可以参考下
    2024-02-02
  • vue如何实现动态的选中状态切换效果

    vue如何实现动态的选中状态切换效果

    这篇文章主要介绍了vue如何实现动态的选中状态切换效果,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2022-04-04
  • 前端配合后端实现Vue路由权限的方法实例

    前端配合后端实现Vue路由权限的方法实例

    一开始我还以为vue的路由只能用在工程化的项目里面,其实不然,下面这篇文章主要给大家介绍了关于前端配合后端实现Vue路由权限的相关资料,文中通过实例代码介绍的非常详细,需要的朋友可以参考下
    2022-05-05
  • Vue图片浏览组件v-viewer用法分析【支持旋转、缩放、翻转等操作】

    Vue图片浏览组件v-viewer用法分析【支持旋转、缩放、翻转等操作】

    这篇文章主要介绍了Vue图片浏览组件v-viewer用法,结合实例形式分析了v-viewer的基本功能与使用方法,包括旋转、缩放、翻转等操作技巧,需要的朋友可以参考下
    2019-11-11
  • 在vue或H5中如何使用复制粘贴功能

    在vue或H5中如何使用复制粘贴功能

    这篇文章主要介绍了在vue或H5中如何使用复制粘贴功能,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2022-09-09
  • vue3.0+element表格获取每行数据代码示例

    vue3.0+element表格获取每行数据代码示例

    这篇文章主要给大家介绍了关于vue3.0+element表格获取每行数据的相关资料,在element-ui中,你可以通过为表格的行绑定单击事件来获取表格中的一行数据,需要的朋友可以参考下
    2023-09-09

最新评论

?


http://www.vxiaotou.com