在线观看www成人影院-在线观看www日本免费网站-在线观看www视频-在线观看操-欧美18在线-欧美1级

0
  • 聊天消息
  • 系統消息
  • 評論與回復
登錄后你可以
  • 下載海量資料
  • 學習在線課程
  • 觀看技術視頻
  • 寫文章/發帖/加入社區
會員中心
創作中心

完善資料讓更多小伙伴認識你,還能領取20積分哦,立即完善>

3天內不再提示

SpringBoot超大文件上傳,實現秒傳

jf_ro2CN3Fa ? 來源:芋道源碼 ? 作者:芋道源碼 ? 2022-11-17 10:30 ? 次閱讀

來源:已賦值
  • 前言
  • 詳細教程
    • 秒傳
    • 分片上傳
    • 斷點續傳
    • 后端進行寫入操作的核心代碼
  • 總結

前言

文件上傳是一個老生常談的話題了,在文件相對比較小的情況下,可以直接把文件轉化為字節流上傳到服務器,但在文件比較大的情況下,用普通的方式進行上傳,這可不是一個好的辦法,畢竟很少有人會忍受,當文件上傳到一半中斷后,繼續上傳卻只能重頭開始上傳,這種讓人不爽的體驗。那有沒有比較好的上傳體驗呢,答案有的,就是下邊要介紹的幾種上傳方式

基于 Spring Boot + MyBatis Plus + Vue & Element 實現的后臺管理系統 + 用戶小程序,支持 RBAC 動態權限、多租戶、數據權限、工作流、三方登錄、支付、短信、商城等功能

  • 項目地址:https://gitee.com/zhijiantianya/ruoyi-vue-pro
  • 視頻教程:https://doc.iocoder.cn/video/

詳細教程

秒傳

1、什么是秒傳

通俗的說,你把要上傳的東西上傳,服務器會先做MD5校驗,如果服務器上有一樣的東西,它就直接給你個新地址,其實你下載的都是服務器上的同一個文件,想要不秒傳,其實只要讓MD5改變,就是對文件本身做一下修改(改名字不行),例如一個文本文件,你多加幾個字,MD5就變了,就不會秒傳了.

2、本文實現的秒傳核心邏輯

a、利用redis的set方法存放文件上傳狀態,其中key為文件上傳的md5,value為是否上傳完成的標志位,

b、當標志位true為上傳已經完成,此時如果有相同文件上傳,則進入秒傳邏輯。如果標志位為false,則說明還沒上傳完成,此時需要在調用set的方法,保存塊號文件記錄的路徑,其中key為上傳文件md5加一個固定前綴,value為塊號文件記錄路徑

分片上傳

1.什么是分片上傳

分片上傳,就是將所要上傳的文件,按照一定的大小,將整個文件分隔成多個數據塊(我們稱之為Part)來進行分別上傳,上傳完之后再由服務端對所有上傳的文件進行匯總整合成原始的文件。

2.分片上傳的場景

1.大文件上傳

2.網絡環境環境不好,存在需要重傳風險的場景

斷點續傳

1、什么是斷點續傳

斷點續傳是在下載或上傳時,將下載或上傳任務(一個文件或一個壓縮包)人為的劃分為幾個部分,每一個部分采用一個線程進行上傳或下載,如果碰到網絡故障,可以從已經上傳或下載的部分開始繼續上傳或者下載未完成的部分,而沒有必要從頭開始上傳或者下載。本文的斷點續傳主要是針對斷點上傳場景。

2、應用場景

斷點續傳可以看成是分片上傳的一個衍生,因此可以使用分片上傳的場景,都可以使用斷點續傳。

3、實現斷點續傳的核心邏輯

在分片上傳的過程中,如果因為系統崩潰或者網絡中斷等異常因素導致上傳中斷,這時候客戶端需要記錄上傳的進度。在之后支持再次上傳時,可以繼續從上次上傳中斷的地方進行繼續上傳。

為了避免客戶端在上傳之后的進度數據被刪除而導致重新開始從頭上傳的問題,服務端也可以提供相應的接口便于客戶端對已經上傳的分片數據進行查詢,從而使客戶端知道已經上傳的分片數據,從而從下一個分片數據開始繼續上傳。

4、實現流程步驟

a、方案一,常規步驟

  • 將需要上傳的文件按照一定的分割規則,分割成相同大小的數據塊;
  • 初始化一個分片上傳任務,返回本次分片上傳唯一標識;
  • 按照一定的策略(串行或并行)發送各個分片數據塊;
  • 發送完成后,服務端根據判斷數據上傳是否完整,如果完整,則進行數據塊合成得到原始文件。

b、方案二、本文實現的步驟

  • 前端(客戶端)需要根據固定大小對文件進行分片,請求后端(服務端)時要帶上分片序號和大小
  • 服務端創建conf文件用來記錄分塊位置,conf文件長度為總分片數,每上傳一個分塊即向conf文件中寫入一個127,那么沒上傳的位置就是默認的0,已上傳的就是Byte.MAX_VALUE 127(這步是實現斷點續傳和秒傳的核心步驟)
  • 服務器按照請求數據中給的分片序號和每片分塊大小(分片大小是固定且一樣的)算出開始位置,與讀取到的文件片段數據,寫入文件。
5、分片上傳/斷點上傳代碼實現

a、前端采用百度提供的webuploader的插件,進行分片。因本文主要介紹服務端代碼實現,webuploader如何進行分片,具體實現可以查看如下鏈接:

http://fex.baidu.com/webuploader/getting-started.html

b、后端用兩種方式實現文件寫入,一種是用RandomAccessFile,如果對RandomAccessFile不熟悉的朋友,可以查看如下鏈接:

https://blog.csdn.net/dimudan2015/article/details/81910690

另一種是使用MappedByteBuffer,對MappedByteBuffer不熟悉的朋友,可以查看如下鏈接進行了解:

https://www.jianshu.com/p/f90866dcbffc

后端進行寫入操作的核心代碼

a、RandomAccessFile實現方式

@UploadMode(mode=UploadModeEnum.RANDOM_ACCESS)
@Slf4j
publicclassRandomAccessUploadStrategyextendsSliceUploadTemplate{

@Autowired
privateFilePathUtilfilePathUtil;

@Value("${upload.chunkSize}")
privatelongdefaultChunkSize;

@Override
publicbooleanupload(FileUploadRequestDTOparam){
RandomAccessFileaccessTmpFile=null;
try{
StringuploadDirPath=filePathUtil.getPath(param);
FiletmpFile=super.createTmpFile(param);
accessTmpFile=newRandomAccessFile(tmpFile,"rw");
//這個必須與前端設定的值一致
longchunkSize=Objects.isNull(param.getChunkSize())?defaultChunkSize*1024*1024
:param.getChunkSize();
longoffset=chunkSize*param.getChunk();
//定位到該分片的偏移量
accessTmpFile.seek(offset);
//寫入該分片數據
accessTmpFile.write(param.getFile().getBytes());
booleanisOk=super.checkAndSetUploadProgress(param,uploadDirPath);
returnisOk;
}catch(IOExceptione){
log.error(e.getMessage(),e);
}finally{
FileUtil.close(accessTmpFile);
}
returnfalse;
}

}

b、MappedByteBuffer實現方式

@UploadMode(mode=UploadModeEnum.MAPPED_BYTEBUFFER)
@Slf4j
publicclassMappedByteBufferUploadStrategyextendsSliceUploadTemplate{

@Autowired
privateFilePathUtilfilePathUtil;

@Value("${upload.chunkSize}")
privatelongdefaultChunkSize;

@Override
publicbooleanupload(FileUploadRequestDTOparam){

RandomAccessFiletempRaf=null;
FileChannelfileChannel=null;
MappedByteBuffermappedByteBuffer=null;
try{
StringuploadDirPath=filePathUtil.getPath(param);
FiletmpFile=super.createTmpFile(param);
tempRaf=newRandomAccessFile(tmpFile,"rw");
fileChannel=tempRaf.getChannel();

longchunkSize=Objects.isNull(param.getChunkSize())?defaultChunkSize*1024*1024
:param.getChunkSize();
//寫入該分片數據
longoffset=chunkSize*param.getChunk();
byte[]fileData=param.getFile().getBytes();
mappedByteBuffer=fileChannel
.map(FileChannel.MapMode.READ_WRITE,offset,fileData.length);
mappedByteBuffer.put(fileData);
booleanisOk=super.checkAndSetUploadProgress(param,uploadDirPath);
returnisOk;

}catch(IOExceptione){
log.error(e.getMessage(),e);
}finally{

FileUtil.freedMappedByteBuffer(mappedByteBuffer);
FileUtil.close(fileChannel);
FileUtil.close(tempRaf);

}

returnfalse;
}

}

c、文件操作核心模板類代碼

@Slf4j
publicabstractclassSliceUploadTemplateimplementsSliceUploadStrategy{

publicabstractbooleanupload(FileUploadRequestDTOparam);

protectedFilecreateTmpFile(FileUploadRequestDTOparam){

FilePathUtilfilePathUtil=SpringContextHolder.getBean(FilePathUtil.class);
param.setPath(FileUtil.withoutHeadAndTailDiagonal(param.getPath()));
StringfileName=param.getFile().getOriginalFilename();
StringuploadDirPath=filePathUtil.getPath(param);
StringtempFileName=fileName+"_tmp";
FiletmpDir=newFile(uploadDirPath);
FiletmpFile=newFile(uploadDirPath,tempFileName);
if(!tmpDir.exists()){
tmpDir.mkdirs();
}
returntmpFile;
}

@Override
publicFileUploadDTOsliceUpload(FileUploadRequestDTOparam){

booleanisOk=this.upload(param);
if(isOk){
FiletmpFile=this.createTmpFile(param);
FileUploadDTOfileUploadDTO=this.saveAndFileUploadDTO(param.getFile().getOriginalFilename(),tmpFile);
returnfileUploadDTO;
}
Stringmd5=FileMD5Util.getFileMD5(param.getFile());

Mapmap=newHashMap<>();
map.put(param.getChunk(),md5);
returnFileUploadDTO.builder().chunkMd5Info(map).build();
}

/**
*檢查并修改文件上傳進度
*/
publicbooleancheckAndSetUploadProgress(FileUploadRequestDTOparam,StringuploadDirPath){

StringfileName=param.getFile().getOriginalFilename();
FileconfFile=newFile(uploadDirPath,fileName+".conf");
byteisComplete=0;
RandomAccessFileaccessConfFile=null;
try{
accessConfFile=newRandomAccessFile(confFile,"rw");
//把該分段標記為true表示完成
System.out.println("setpart"+param.getChunk()+"complete");
//創建conf文件文件長度為總分片數,每上傳一個分塊即向conf文件中寫入一個127,那么沒上傳的位置就是默認0,已上傳的就是Byte.MAX_VALUE127
accessConfFile.setLength(param.getChunks());
accessConfFile.seek(param.getChunk());
accessConfFile.write(Byte.MAX_VALUE);

//completeList檢查是否全部完成,如果數組里是否全部都是127(全部分片都成功上傳)
byte[]completeList=FileUtils.readFileToByteArray(confFile);
isComplete=Byte.MAX_VALUE;
for(inti=0;i//與運算,如果有部分沒有完成則isComplete不是Byte.MAX_VALUE
isComplete=(byte)(isComplete&completeList[i]);
System.out.println("checkpart"+i+"complete?:"+completeList[i]);
}

}catch(IOExceptione){
log.error(e.getMessage(),e);
}finally{
FileUtil.close(accessConfFile);
}
booleanisOk=setUploadProgress2Redis(param,uploadDirPath,fileName,confFile,isComplete);
returnisOk;
}

/**
*把上傳進度信息存進redis
*/
privatebooleansetUploadProgress2Redis(FileUploadRequestDTOparam,StringuploadDirPath,
StringfileName,FileconfFile,byteisComplete){

RedisUtilredisUtil=SpringContextHolder.getBean(RedisUtil.class);
if(isComplete==Byte.MAX_VALUE){
redisUtil.hset(FileConstant.FILE_UPLOAD_STATUS,param.getMd5(),"true");
redisUtil.del(FileConstant.FILE_MD5_KEY+param.getMd5());
confFile.delete();
returntrue;
}else{
if(!redisUtil.hHasKey(FileConstant.FILE_UPLOAD_STATUS,param.getMd5())){
redisUtil.hset(FileConstant.FILE_UPLOAD_STATUS,param.getMd5(),"false");
redisUtil.set(FileConstant.FILE_MD5_KEY+param.getMd5(),
uploadDirPath+FileConstant.FILE_SEPARATORCHAR+fileName+".conf");
}

returnfalse;
}
}
/**
*保存文件操作
*/
publicFileUploadDTOsaveAndFileUploadDTO(StringfileName,FiletmpFile){

FileUploadDTOfileUploadDTO=null;

try{

fileUploadDTO=renameFile(tmpFile,fileName);
if(fileUploadDTO.isUploadComplete()){
System.out
.println("uploadcomplete!!"+fileUploadDTO.isUploadComplete()+"name="+fileName);
//TODO保存文件信息到數據庫

}

}catch(Exceptione){
log.error(e.getMessage(),e);
}finally{

}
returnfileUploadDTO;
}
/**
*文件重命名
*
*@paramtoBeRenamed將要修改名字的文件
*@paramtoFileNewName新的名字
*/
privateFileUploadDTOrenameFile(FiletoBeRenamed,StringtoFileNewName){
//檢查要重命名的文件是否存在,是否是文件
FileUploadDTOfileUploadDTO=newFileUploadDTO();
if(!toBeRenamed.exists()||toBeRenamed.isDirectory()){
log.info("Filedoesnotexist:{}",toBeRenamed.getName());
fileUploadDTO.setUploadComplete(false);
returnfileUploadDTO;
}
Stringext=FileUtil.getExtension(toFileNewName);
Stringp=toBeRenamed.getParent();
StringfilePath=p+FileConstant.FILE_SEPARATORCHAR+toFileNewName;
FilenewFile=newFile(filePath);
//修改文件名
booleanuploadFlag=toBeRenamed.renameTo(newFile);

fileUploadDTO.setMtime(DateUtil.getCurrentTimeStamp());
fileUploadDTO.setUploadComplete(uploadFlag);
fileUploadDTO.setPath(filePath);
fileUploadDTO.setSize(newFile.length());
fileUploadDTO.setFileExt(ext);
fileUploadDTO.setFileId(toFileNewName);

returnfileUploadDTO;
}
}

基于 Spring Cloud Alibaba + Gateway + Nacos + RocketMQ + Vue & Element 實現的后臺管理系統 + 用戶小程序,支持 RBAC 動態權限、多租戶、數據權限、工作流、三方登錄、支付、短信、商城等功能

  • 項目地址:https://gitee.com/zhijiantianya/yudao-cloud
  • 視頻教程:https://doc.iocoder.cn/video/

總結

在實現分片上傳的過程,需要前端和后端配合,比如前后端的上傳塊號的文件大小,前后端必須得要一致,否則上傳就會有問題。其次文件相關操作正常都是要搭建一個文件服務器的,比如使用fastdfs、hdfs等。

本示例代碼在電腦配置為4核內存8G情況下,上傳24G大小的文件,上傳時間需要30多分鐘,主要時間耗費在前端的md5值計算,后端寫入的速度還是比較快。如果項目組覺得自建文件服務器太花費時間,且項目的需求僅僅只是上傳下載,那么推薦使用阿里的oss服務器,其介紹可以查看官網:

https://help.aliyun.com/product/31815.html

阿里的oss它本質是一個對象存儲服務器,而非文件服務器,因此如果有涉及到大量刪除或者修改文件的需求,oss可能就不是一個好的選擇。

文末提供一個oss表單上傳的鏈接demo,通過oss表單上傳,可以直接從前端把文件上傳到oss服務器,把上傳的壓力都推給oss服務器:

https://www.cnblogs.com/ossteam/p/4942227.html



審核編輯 :李倩


聲明:本文內容及配圖由入駐作者撰寫或者入駐合作網站授權轉載。文章觀點僅代表作者本人,不代表電子發燒友網立場。文章及其配圖僅供工程師學習之用,如有內容侵權或者其他違規問題,請聯系本站處理。 舉報投訴
  • 服務器
    +關注

    關注

    12

    文章

    9160

    瀏覽量

    85415
  • spring
    +關注

    關注

    0

    文章

    340

    瀏覽量

    14343
  • Boot
    +關注

    關注

    0

    文章

    149

    瀏覽量

    35837
  • SpringBoot
    +關注

    關注

    0

    文章

    173

    瀏覽量

    178

原文標題:SpringBoot超大文件上傳,實現秒傳

文章出處:【微信號:芋道源碼,微信公眾號:芋道源碼】歡迎添加關注!文章轉載請注明出處。

收藏 人收藏

    評論

    相關推薦

    如何使用SFTP傳輸大文件

    在當今的數字化時代,大文件傳輸變得越來越常見。無論是企業數據遷移、遠程備份還是內容分發,都需要一種既安全又高效的文件傳輸方式。SFTP作為一種基于SSH的文件傳輸協議,提供了一種加密的傳輸方式,確保
    的頭像 發表于 11-13 14:11 ?676次閱讀

    P2Link是什么?——免費讓你體驗高效智聯的新方式

    P2Link 是一個非常方便、簡單的工具,可以幫你直接傳輸大文件,或者讓別人訪問你內網中的服務。它不像傳統工具需要復雜的配置,也沒有上傳到云端的煩惱。特別適合那些臨時共享文件
    的頭像 發表于 10-31 14:11 ?187次閱讀

    如何修改buildroot和debian文件系統

    本文檔主要介紹在沒有編譯環境的情況下,如何修改buildroot和debian文件系統方法,如在buildroot文件系統中添加文件、修改目錄等文件操作,在debian
    的頭像 發表于 07-22 17:46 ?485次閱讀
    如何修改buildroot和debian<b class='flag-5'>文件</b>系統

    如何實現Python復制文件操作

    Python 中有許多“開蓋即食”的模塊(比如 os,subprocess 和 shutil)以支持文件 I/O 操作。在這篇文章中,你將會看到一些用 Python 實現文件復制的特殊方法。下面我們開始學習這九種不同的方法來
    的頭像 發表于 07-18 14:53 ?422次閱讀

    請問AT支持UDP透嗎?

    做了測試,SDK V1.00 AT文件燒寫后,設定為TCP時,透是可以了, 但是設定為UDP后,透傳出現發送不成功,但是接收是可以的,是不是AT不支持UDP透呢? 測試的指令如下
    發表于 07-17 08:28

    請問SDK固件如何實現網絡透

    您好,想請問一下,我們現在一個項目需要用SDK來實現網路數據透,但是調用 官方API :espconn_send()會自動把HTTP報文頭部給添加上去了,沒有有辦法實現像AT固件一樣直傳?
    發表于 07-10 06:59

    兩臺電腦怎么文件?干貨分享教程

    當需要在兩臺電腦之間傳輸文件時,有多種方便的方法可供選擇,以下是一些常見的方式及教程: ? 使用局域網共享 : 確保兩臺電腦連接在同一個局域網內。 在其中一臺電腦上,設置要共享的文件夾。右鍵點擊
    的頭像 發表于 07-03 14:39 ?671次閱讀

    esp32如何一次性讀取大文件數據?

    esp32沒有提供數據庫讀寫的例子,最近有個大文件,無法一次性讀出,請問,怎么讀取,json中一部分json數組。然后修改完了以后,在寫入進去?
    發表于 06-25 06:52

    使用ESP32-S3開發板http post請求發送SD卡上的大文件,如何循環邊讀取文件邊分塊發送文件呢?

    您和,我準備使用ESP32-S3開發板http post請求發送SD卡上的大文件,但是使用esp_http_client_set_post_field的buffer太小,內存不能一次性申請太大,請問
    發表于 06-06 06:19

    鴻蒙原生應用元服務開發-Web上傳文件

    Web組件支持前端頁面選擇文件上傳功能,應用開發者可以使用onShowFileSelector()接口來處理前端頁面文件上傳的請求。 下面的示例中,當用戶在前端頁面點擊
    發表于 05-08 11:17

    怎么解決需要在國外網站上傳文件打開和上傳速度慢的問題。

    隨著全球化的深入發展,企業面臨的挑戰之一是在國際環境中保持高效的數據傳輸和通信。上傳文件至國外網站時,傳統的互聯網連接方式往往無法滿足企業對于速度和穩定性的需求,這不僅拖慢了工作進度,還可
    的頭像 發表于 04-28 17:01 ?319次閱讀

    Android版Gemini新增上傳PDF及選取文本等功能

    盡管安卓版 Gemini 當前只支持圖片上傳,但最新的版本已出現了上傳 PDF 等文件的代碼,這意味著它有可能具備解析文檔內容的能力,只是此功能暫未啟用。
    的頭像 發表于 04-24 14:23 ?533次閱讀

    stm32f303rc usb自定義hid下位機只要上傳2包數據,上位機就無法上傳了的原因?

    只要下位機上傳2次數據,不連續,分開,哪怕每次數據只一個字節,這時候,上位機下傳數據立刻失敗,下位機仍然能夠上傳數據。 各位同仁可有指點方向的嗎? 謝謝!
    發表于 03-29 06:06

    激光焊接機如何實現高精度焊接

    編輯:鐳拓激光納激光焊接機實現高精度焊接主要依賴于先進的激光技術和精確的控制系統。以下是鐳拓小編為大家總結的納激光焊接機實現高精度焊接的幾點關鍵因素:1.激光技術:納
    的頭像 發表于 01-29 15:38 ?981次閱讀
    納<b class='flag-5'>秒</b>激光焊接機如何<b class='flag-5'>實現</b>高精度焊接

    如何通過透網關實現PLC的遠程控制?

    【技術分享】遠程透網關-單網口快速實現三菱 FX5U PLC程序遠程上下載
    的頭像 發表于 01-19 17:46 ?1184次閱讀
    如何通過透<b class='flag-5'>傳</b>網關<b class='flag-5'>實現</b>PLC的遠程控制?
    主站蜘蛛池模板: 色www 永久免费网站| 麦克斯奥特曼免费观看| 精品国产一区二区三区国产馆| 亚洲精品视频免费| 激情伦成人综合小说| 特级片在线观看| 丁香婷婷视频| 亚洲成a人伦理| 天堂亚洲网| www.色婷婷.com| 日韩操穴| 四虎影院com| 亚洲加勒比在线| 中文字幕一区二区三区5566| 色爱综合网欧美| 天天碰夜夜| 精品成人| 国产一二三区精品| 1000又爽又黄禁片| 四虎影院免费在线| 一本到视频在线| 欧美高清成人| 寄宿日记免费看| 亚洲成成品网站有线| 色域综合| 深夜影院一级毛片| 在线观看精品视频看看播放| 国产香蕉75在线播放| 涩涩色中文综合亚洲| 日本免费高清| 男人午夜影院| 五月天精品在线| 午夜亚洲国产| 最色成人网| 999色综合| 男女性生动态免费视频| 国模沟沟一区二区三区| 欧美人与z0zoxxxx| 欧美成人综合在线| 国产精品a在线观看香蕉| 四虎最新在线|