异步上传文件显示进度条

2 篇文章 0 订阅

异步上传文件显示进度条

原文地址:异步上传文件显示进度条

问题

我们在写网站时难免会遇到需要上传文件的场景,但当上传大文件时比如5个G的文件直接用表单直接提交文件会出现页面卡顿、未响应等影响用户体验的情况,而且用户也不知道文件上传的进度,他就会认为是不是网页卡死了从而直接关闭页面或者刷新页面而导致上传失败

解决

那么针对上述的问题我们有没有什么好的解决办法呢?答案肯定是有的

我们可以利用异步+监听来帮我们解决这个问题

创建一个FormData对象

  • 我们可以手动创建一个表单对象来存放我们要上传的文件,代码如下:
let fd = new FormData();
fd.append("file", fs); //fs 为要上传的文件

利用XMLHttpRequest发送异步请求去提交表单

  • 创建一个XMLHttpRequest对象
  • 将上步创建的表单对象放入request body
  • 开始上传
  • 成功回调
let xhr = new XMLHttpRequest();
// /upload为目标url,换成你自己的
xhr.open("POST", "/upload", true);
xhr.send(fd);
xhr.onloadend = function () {
  alert("上传成功")
}

加入监听

  • 完成上述操作我们已经可以异步上传,但异步上传时用户还不知道上传的进度,所以我们需要在上传之前加上监听
let xhr = new XMLHttpRequest();
// /upload为目标url,换成你自己的
xhr.open("POST", "/upload", true);
// uploadProgress为监听的回调
xhr.upload.addEventListener("progress", uploadProgress, false);
xhr.send(fd);
xhr.onloadend = function () {
  alert("上传成功")
}
  • uploadProgress中获取上传进度
function uploadProgress(evt) {
  if (evt.lengthComputable) {
    // evt.loaded:文件上传的大小 evt.total:文件总的大小
    console.log(evt.loaded);
    console.log(evt.total);
    let percentComplete = Math.round((evt.loaded) * 100 / evt.total);
    consolo.log("进度:"+percentComplete+"%")
  }
}

完整代码

效果

我这里上传一个5G多的文件为例

https://gitee.com/kevinlu98/imgbed/raw/master/20220125/4DHXV2WXRfFc.gif

前端代码

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>Title</title>
    <link rel="stylesheet" href="https://cdn.jsdelivr.net/npm/bootstrap@4.6.1/dist/css/bootstrap.min.css">
</head>
<body>
<div class="card">
    <h5 class="card-header">上传演示</h5>
    <div class="card-body">
        <p>
            <button type="button" id="upload-btn" class="btn btn-primary">上传文件</button>
        </p>
        <h5 class="card-title">上传进度</h5>
        <div class="progress">
            <div id="upload-progress" class="progress-bar" role="progressbar" aria-valuenow="0" aria-valuemin="0"
                 aria-valuemax="100"></div>
        </div>
    </div>
</div>
<div id="upload-box" class="hide" style="display: none">
    <input type="file">
</div>
<script src="https://cdn.bootcdn.net/ajax/libs/jquery/3.6.0/jquery.min.js"></script>
<script src="https://cdn.jsdelivr.net/npm/jquery@3.5.1/dist/jquery.slim.min.js"></script>
<script src="https://cdn.jsdelivr.net/npm/bootstrap@4.6.1/dist/js/bootstrap.bundle.min.js"></script>
<script>
    $(function () {
        $("#upload-btn").on('click', function () {
            $("#upload-box input[type=file]").click();
        })
        let fs = {}
        $("#upload-box").on('change', 'input[type=file]', function () {
            fs = this.files[0]
            $("#upload-box").html('<input type="file">');
            doUpload()
        })

        function doUpload() {
            let xhr = new XMLHttpRequest();
            let fd = new FormData();
            fd.append("file", fs);
            xhr.open("POST", "/upload", true);// 上传目标
            xhr.upload.addEventListener("progress", uploadProgress, false);
            xhr.send(fd);
            xhr.onloadend = function () {
                alert("上传成功")
            }
        }

        function uploadProgress(evt) {
            if (evt.lengthComputable) {
                // evt.loaded:文件上传的大小 evt.total:文件总的大小
                console.log(evt.loaded);
                console.log(evt.total);
                let percentComplete = Math.round((evt.loaded) * 100 / evt.total);
                $("#upload-progress").width(percentComplete + "%").attr("aria-valuenow",percentComplete)
            }
        }
    })
</script>
</body>
</html>

后端Java代码

我就以后端程序为springboot为例,写了一个简单的上传文件的Controller

package com.example.filedemo.controller;

import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.PostMapping;
import org.springframework.web.bind.annotation.RequestParam;
import org.springframework.web.bind.annotation.ResponseBody;
import org.springframework.web.multipart.MultipartFile;
import java.io.File;
import java.io.IOException;

@Controller
public class UploadController {
    private static final Logger LOGGER = LoggerFactory.getLogger(UploadController.class);

    @PostMapping("/upload")
    @ResponseBody
    public String upload(@RequestParam("file") MultipartFile file) {
        if (file.isEmpty()) {
            return "上传失败,请选择文件";
        }

        String fileName = file.getOriginalFilename();
        String filePath = "./temp/";
        File dest = new File(filePath + fileName);
        try {
            file.transferTo(dest);
            LOGGER.info("上传成功");
            return "上传成功";
        } catch (IOException e) {
            LOGGER.error(e.getMessage(), e);
        }
        return "上传失败!";
    }


}

  • 2
    点赞
  • 14
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
前端开发中,我们经常会遇到需要上传文件的场景,而 FormData 就是一种能够方便实现文件上传的技术手段。而后端 Qt 则是一种强大的跨平台开发框架,具有良好的可移植性和高效性能,能够轻松地实现各种类型的应用开发。 在前端使用 FormData 进行文件上传的过程中,需要使用 JavaScript API 来获取文件并将其添加到 FormData 中。在上传文件时,我们可以通过 XMLHttpRequest(XHR)对象或者 Fetch API 对 formData 对象进行异步请求发送。而后端 Qt 中,则需要使用相应的网络模块来处理收到的文件上传请求,同时需要完成文件的存储和相关的逻辑处理。 首先,在进行文件上传前,我们需要明确前后端传输数据的格式和类型,并且确保前后端约定的接口名称和参数格式一致。对于文件上传,我们还需要注意文件格式和大小的限制,以及如何处理上传失败、重复上传等问题。 其次,前端 FormData 对象在上传文件时需要注意以下几点: 1.通过使用append() 方法将文件数据添加到 FormData 中; 2.设置 xhr 发送的请求头,包括 Content-Type 等参数; 3.监听 xhr 上传进度,根据上传进度显示进度条等信息; 4.对于上传失败的情况,需要进行错误处理并给出相应的提示。 最后,在后端 Qt 中,我们需要使用相应的文件上传接口处理收到的请求。同样需要注意以下几点: 1.获取 FormData 中的文件数据,并保存到指定的目录下; 2.对上传的文件进行必要的内容验证和格式检查; 3.根据实际需求,进行文件命名、保存路径等调整; 4.根据上传的文件数据完成后续的逻辑处理和响应。 通过以上步骤,前端的 FormData 文件上传和后端 Qt 的文件处理可以得到成功的协同。同时,在实际项目中,我们还需要考虑安全、兼容性等方面的因素,确保系统的可靠性和稳定性。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值