Java实现浏览器端大文件分片上传实例

javaweb上传文件

上传文件的jsp中的部分

上传文件同样可以使用form表单向后端发请求,也可以使用 ajax向后端发请求

    1. 通过form表单向后端发送请求

         <form id="postForm" action="${pageContext.request.contextPath}/UploadServlet" method="post" enctype="multipart/form-data">

            <div class="bbxx wrap">

                <input type="text" id="side-profile-name" name="username" class="form-control">

                <input type="file" id="example-file-input" name="avatar">

                <button type="submit" class="btn btn-effect-ripple btn-primary">Save</button>

            </div>

        </form>

改进后的代码不需要form标签,直接由控件来实现。开发人员只需要关注业务逻辑即可。JS中已经帮我们封闭好了

通过监控工具可以看到控件提交的数据,非常的清晰,调试也非常的简单。

 

2. 通过ajax向后端发送请求

        1.

            $.ajax({ 

                 url : "${pageContext.request.contextPath}/UploadServlet", 

                 type : "POST", 

                 data : $( '#postForm').serialize(), 

                 success : function(data) { 

                      $( '#serverResponse').html(data); 

                 }, 

                 error : function(data) { 

                      $( '#serverResponse').html(data.status + " : " + data.statusText + " : " + data.responseText); 

                 } 

            }); 

ajax分为两部分,一部分是初始化,文件在上传前通过AJAX请求通知服务端进行初始化操作

 

在文件上传完后向服务器发送通知

 

这里需要处理一个MD5秒传的逻辑,当服务器存在相同文件时,不需要用户再上传,而是直接通知用户秒传

这里可以看到秒传的逻辑是非常 简单的,并不是特别的复杂。

            var form = new FormData();

            form.append("username","zxj");

            form.append("avatar",file);

            //var form = new FormData($("#postForm")[0]);

            $.ajax({

                url:"${pageContext.request.contextPath}/UploadServlet",

                type:"post",

                data:form,

                processData:false,

                contentType:false,

                success:function(data){

         

                    console.log(data);

                }

            });

java部分

文件初始化的逻辑

 

接收文件块数据,在这个逻辑中我们接收文件块数据。控件对数据进行了优化,可以方便调试。如果用监控工具可以看到控件提交的数据。

 

注:

1. 上面的java部分的代码可以直接使用,只需要将上传的图片路径及收集数据并将数据写入到数据库即可

2. 上面上传文件使用到了字节流,其实还可以使用别的流,这个需要读者自己在下面完善测试

3. BeanUtils是一个工具 便于将实体对应的属性赋给实体

4. 上传文件不能使用 request.getParameter("")获取参数了,而是直接将request解析,通过判断每一项是文件还是非文件,然后进行相应的操作(文件的话就是用流来读取,非文件的话,暂时保存到一个map中。)

后端代码逻辑大部分是相同的,目前能够支持MySQL,Oracle,SQL。在使用前需要配置一下数据库,可以参考我写的这篇文章:http://blog.ncmem.com/wordpress/2019/08/12/java-http%E5%A4%A7%E6%96%87%E4%BB%B6%E6%96%AD%E7%82%B9%E7%BB%AD%E4%BC%A0%E4%B8%8A%E4%BC%A0/

欢迎入群一起讨论:374992201

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
好的,下面是一个基于前后的多文件和大文件分片的示例代码: 首先是HTML部分,使用input标签实现文件选择和上按钮: ```html <input type="file" multiple id="file-input"> <button onclick="upload()">上</button> ``` 接下来是JavaScript部分,实现了多文件和大文件分片功能。其中,upload函数是上按钮的点击事件处理函数。 ```javascript function upload() { const fileInput = document.getElementById('file-input'); const files = fileInput.files; const chunkSize = 1024 * 1024; // 分片大小为1MB const totalChunks = Math.ceil(files.size / chunkSize); // 计算总块数 const formData = new FormData(); formData.append('totalChunks', totalChunks); // 上总块数 for (let i = 0; i < files.length; i++) { const file = files[i]; const chunks = Math.ceil(file.size / chunkSize); // 计算当前文件的总块数 formData.append('file', file); // 添加文件数据到FormData对象 for (let j = 0; j < chunks; j++) { const start = j * chunkSize; const end = Math.min(start + chunkSize, file.size); const chunk = file.slice(start, end); formData.append('chunkIndex', j); formData.append('chunkData', chunk); // 使用Fetch API进行上 fetch('/upload', { method: 'POST', body: formData }) .then(response => response.json()) .then(data => console.log(data)) .catch(error => console.error(error)); } } } ``` 以上代码中,我们先获取了input标签中选择的所有文件,然后对每个文件进行分片。每个分片的大小默认为1MB,可以根据实际情况进行调整。接着使用FormData对象构造包含分片数据的表单数据,然后使用Fetch API进行上。上的URL为`/upload`,可以根据实际情况进行调整。 在服务,我们需要接收上分片数据,并在接收完所有分片后进行合并。以下是Node.js的示例代码: ```javascript const express = require('express'); const app = express(); const fs = require('fs'); const path = require('path'); const bodyParser = require('body-parser'); app.use(bodyParser.json()); app.use(bodyParser.urlencoded({ extended: true })); app.post('/upload', (req, res) => { const totalChunks = req.body.totalChunks; const fileName = req.body.file.name; const chunkIndex = req.body.chunkIndex; const chunkData = req.body.chunkData; const filePath = path.join(__dirname, fileName + '.' + chunkIndex); // 将分片数据写入文件 fs.writeFile(filePath, chunkData, (err) => { if (err) { console.error(err); res.status(500).send('Failed to write chunk data'); return; } // 如果是最后一个分片,则进行合并 if (chunkIndex === totalChunks - 1) { const writeStream = fs.createWriteStream(fileName); for (let i = 0; i < totalChunks; i++) { const chunkPath = path.join(__dirname, fileName + '.' + i); const chunkData = fs.readFileSync(chunkPath); writeStream.write(chunkData); fs.unlinkSync(chunkPath); // 删除已合并的分片文件 } writeStream.end(() => { res.send('Upload success'); }); } else { res.send('Chunk uploaded'); } }); }); app.listen(3000, () => { console.log('Server started at http://localhost:3000'); }); ``` 以上代码中,我们使用Express框架接收POST请求,并从请求体中获取分片数据、文件名、当前分片索引、总块数等信息。然后将分片数据写入文件,如果是最后一个分片,则进行合并,最终得到完整的文件。注意,分片文件的命名规则为`文件名.分片索引`,合并完成后需要删除已合并的分片文件。 这就是一个基于前后的多文件和大文件分片实现示例,希望能对你有所帮助。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值