最近在公司的开发任务中,有一个大文件需要分片上传的功能要求,由于之前没有做过这个功能,看了一下网上的博客,有看到生成一些小文件再上传的,感觉不是很好,所以在这记录一下最后的代码,希望能跟大家交流一下
分块方法
/**
* @param offset 偏移量
* @param file 分块文件
* @param blockSize 每块的大小
* @return 这一片的数据
*/
private byte[] getBlock(long offset, File file, int blockSize) {
byte[] result = new byte[blockSize];
try (RandomAccessFile accessFile = new RandomAccessFile(file, "r")) {
accessFile.seek(offset);
int readSize = accessFile.read(result);
if (readSize == -1) {
return null;
} else if (readSize == blockSize) {
return result;
} else {
byte[] byteArray = new byte[readSize];
System.arraycopy(result, 0, byteArray, 0, readSize);
return byteArray;
}
} catch (IOException e) {
e.printStackTrace();
}
return null;
}
private void uploadFile(String filePath){
File file = new File(filePath);
if (file .exists()){
//md5值是后台接口需要
String md5 = getFileMD5(file );
//计算文件分片的总块数
long totalBlock = sourceFile.length() / PART_SIZE + (sourceFile.length() % PART_SIZE > 0 ? 1 : 0);
//不要怀疑Math.max(totalBlock,1)的用法,因为我要上传0长度但有内容的文件(真的有!)
uploadFile(filePath,file.getName(),file.length(),Math.max(totalBlock,1),1,md5);
} else {
//不存在则回调错误方法
msg.append("file not exists");
onError();
}
}
private void uploadFile(String filePath, String fileName, long totalSize, long totalBlock, int currentBlock, String md5){
File file = new File(filePath);
byte[] fileStream = getBlock((currentBlock-1)*PART_SIZE,file,PART_SIZE);
if (fileStream == null) {
msg.append("getBlock error");
onError();
return;
}
MultipartBody.Builder builder = new MultipartBody.Builder();
builder.setType(MultipartBody.FORM);
builder.addFormDataPart("totalSize", String.valueOf(totalSize))
.addFormDataPart("totalBlock", String.valueOf(totalBlock))
.addFormDataPart("currentBlock", String.valueOf(currentBlock))
.addFormDataPart("md5",md5)
.addFormDataPart("type","file");
RequestBody body = RequestBody.create(MultipartBody.FORM, fileStream);
builder.addFormDataPart("file", fileName, body);
RetrofitUtil.upload(token,builder.build()).enqueue(new Callback<UploadSuccess>() {
@Override
public void onResponse(Call<UploadSuccess> call, Response<UploadSuccess> response) {
UploadSuccess success = response.body();
if (success != null) {
if (success.isStatus()) {
if (currentBlock == totalBlock) {
onSuccess();
} else {
// 在回调里面上传下一分片,比直接循环更加友好
uploadFile(filePath,fileName,totalSize,totalBlock,currentBlock+1,md5,callBack);
}
} else {
msg.append(success.getMsg());
onError();
}
}
}
@Override
public void onFailure(Call<UploadSuccess> call, Throwable t) {
msg.append(t.getMessage());
t.printStackTrace();
onError();
}
});
}
@POST("api/upload")
Call<UploadSuccess> upload(@Header("Authorization")String token, @Body RequestBody requestBody);
本文链接https://blog.csdn.net/weixin_44337681/article/details/112900048
未经允许,请勿转载!
文章浏览阅读1.8k次。摘自: [ORACLE EBS 入门及供应链核心系统详解教程] (书籍)EBS基础功能架构(13个核心模块,业财一体化)业务运营管理,价值增值财务会计管理,价值实现应用架构Finance财务,资金流Accounting财务管理Bisuness业务,实物流核心业务,与财务高度集成;PUR、INV、制造、订单履行等间接业务,or专业业务,为核心业务提供支持;HR..._ebs r12
文章浏览阅读838次。转载:https://blog.csdn.net/ccecwg/article/details/39546307_java date timestamp区别
文章浏览阅读1.4k次。1、我们先了解一下原生js中的选择器ID选择器(在整个文档中获取id为xxx的元素)document.getElementId([ID]);类名选择器(在整个文档中或者在指定上下文中获取类名为xxx的元素)document.getElementsByClassName(' ');[context].getElementsByClassName(' ');标签名选择器(在整个文档中或者..._原声js实现jq元素选择器
文章浏览阅读1.2k次,点赞3次,收藏4次。通常查询时会对整个数据库查询,而这带来了大量的开销,因此引入了partition的概念,在建表的时候通过设置partition的字段, 会根据该字段对数据分区存放,更具体的说是存放在不同的文件夹,这样通过指定设置Partition的字段条件查询时可以减少大量的开销。1)partition by [key..] order by [key..]只能在窗口函数中使用,而distribute by [key...] sort by [key...]在窗口函数和select中都可以使用。_partition by distribute by
文章浏览阅读7.3k次。Private SQL Area A private SQL area holds information about a parsed SQLstatement and other session-specific information for processing. When a serverprocess executes SQL or PL/SQL code, the process_c# cursor
文章浏览阅读616次。近日在用ListView中的一些注意点,和公用代码,整理如下1.ListView.Items.Clear而不是ListView.Clear一般如果ListView是动态填充的,我们在填充之前都会先进行清理。但需要注意一下,我们是清理Items,如果去直接Clear整个ListView,就连原先定义好的列都没有了2.给ListView绑定数据ListView并不能直接_listview的使用——购物商城实验心得
文章浏览阅读110次。如果没有用来读取注解的方法和工作,那么注解也就不会比注释更有用处了。使用注解的过程中,很重要的一部分就是创建于使用注解处理器。Java SE5扩展了反射机制的API,以帮助程序员快速的构造自定义注解处理器。注解处理器类库(java.lang.reflect.AnnotatedElement):Java使用Annotation接口来代表程序元素前面的注解,该接口是所有Annotation类型的父接口..._java注解处理器作用
文章浏览阅读1.8k次,点赞27次,收藏28次。全国职业技能大赛高职组(最新职业院校技能大赛_大数据应用开发样题解析-模块B:数据采集-任务一:离线数据采集-程序员宅基地。_大数据 国赛 样题
文章浏览阅读926次。本系统分为管理员和注册用户两个角色,主要有疫情新闻、疫情案例介绍、健康信息申报、行程信息申报、就医流程介绍、举报、在线留言、用户管理、信息统计等模块。用户需要先注册成为会员,成功登录后,可以查看网站发布的疫情新闻,可以查看疫情相关病例介绍,有助于疫情防范,还可以查看网站发布的重大疫情案例,了解疫情的发展状况,出行时候好做好防护,同时通过网站可以上报健康信息,以及上报行程信息,方便社区了解自己的出行情况;网站还发布了疫情状态下的就医流程,方便大家就医时候做好准备;同时网站还提供了举报功能,如果发现外来人员或_ssm+微信小程序
文章浏览阅读296次,点赞3次,收藏9次。本节关键字:Linux、centos、串口、U盘、共享文件夹本节相关指令:echo、cat、mkdir、mount
文章浏览阅读1.3k次,点赞45次,收藏29次。本篇主题为: 解密C++新特性:内联函数、auto关键字和基于范围的for循环。
文章浏览阅读774次,点赞4次,收藏11次。1、浏览器常见的报错信息与含义2、304与204的区别,http缓存,强缓存,协商缓存3、浏览器从输入地址到渲染,经历了什么状态?4、vue的界面渲染,经过哪些过程(生命周期)5、三次握手,四次挥手6、重排与重绘7、用css实现一个三角形8、常见的flex布局,有哪些功能9、用css实现一个水平垂直居中10、null与undefined的区别11、虚拟dom12、深拷贝与浅拷贝13、es6新增的功能15、async await 与promise。_今年的前端面试难不难