色哟哟视频在线观看-色哟哟视频在线-色哟哟欧美15最新在线-色哟哟免费在线观看-国产l精品国产亚洲区在线观看-国产l精品国产亚洲区久久

0
  • 聊天消息
  • 系統(tǒng)消息
  • 評(píng)論與回復(fù)
登錄后你可以
  • 下載海量資料
  • 學(xué)習(xí)在線課程
  • 觀看技術(shù)視頻
  • 寫(xiě)文章/發(fā)帖/加入社區(qū)
會(huì)員中心
創(chuàng)作中心

完善資料讓更多小伙伴認(rèn)識(shí)你,還能領(lǐng)取20積分哦,立即完善>

3天內(nèi)不再提示

SpringBoot超大文件上傳,實(shí)現(xiàn)秒傳

jf_ro2CN3Fa ? 來(lái)源:芋道源碼 ? 作者:芋道源碼 ? 2022-11-17 10:30 ? 次閱讀

來(lái)源:已賦值
  • 前言
  • 詳細(xì)教程
    • 秒傳
    • 分片上傳
    • 斷點(diǎn)續(xù)傳
    • 后端進(jìn)行寫(xiě)入操作的核心代碼
  • 總結(jié)

前言

文件上傳是一個(gè)老生常談的話題了,在文件相對(duì)比較小的情況下,可以直接把文件轉(zhuǎn)化為字節(jié)流上傳到服務(wù)器,但在文件比較大的情況下,用普通的方式進(jìn)行上傳,這可不是一個(gè)好的辦法,畢竟很少有人會(huì)忍受,當(dāng)文件上傳到一半中斷后,繼續(xù)上傳卻只能重頭開(kāi)始上傳,這種讓人不爽的體驗(yàn)。那有沒(méi)有比較好的上傳體驗(yàn)?zāi)兀鸢赣械模褪窍逻呉榻B的幾種上傳方式

基于 Spring Boot + MyBatis Plus + Vue & Element 實(shí)現(xiàn)的后臺(tái)管理系統(tǒng) + 用戶小程序,支持 RBAC 動(dòng)態(tài)權(quán)限、多租戶、數(shù)據(jù)權(quán)限、工作流、三方登錄、支付、短信、商城等功能

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

詳細(xì)教程

秒傳

1、什么是秒傳

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

2、本文實(shí)現(xiàn)的秒傳核心邏輯

a、利用redis的set方法存放文件上傳狀態(tài),其中key為文件上傳的md5,value為是否上傳完成的標(biāo)志位,

b、當(dāng)標(biāo)志位true為上傳已經(jīng)完成,此時(shí)如果有相同文件上傳,則進(jìn)入秒傳邏輯。如果標(biāo)志位為false,則說(shuō)明還沒(méi)上傳完成,此時(shí)需要在調(diào)用set的方法,保存塊號(hào)文件記錄的路徑,其中key為上傳文件md5加一個(gè)固定前綴,value為塊號(hào)文件記錄路徑

分片上傳

1.什么是分片上傳

分片上傳,就是將所要上傳的文件,按照一定的大小,將整個(gè)文件分隔成多個(gè)數(shù)據(jù)塊(我們稱(chēng)之為Part)來(lái)進(jìn)行分別上傳,上傳完之后再由服務(wù)端對(duì)所有上傳的文件進(jìn)行匯總整合成原始的文件。

2.分片上傳的場(chǎng)景

1.大文件上傳

2.網(wǎng)絡(luò)環(huán)境環(huán)境不好,存在需要重傳風(fēng)險(xiǎn)的場(chǎng)景

斷點(diǎn)續(xù)傳

1、什么是斷點(diǎn)續(xù)傳

斷點(diǎn)續(xù)傳是在下載或上傳時(shí),將下載或上傳任務(wù)(一個(gè)文件或一個(gè)壓縮包)人為的劃分為幾個(gè)部分,每一個(gè)部分采用一個(gè)線程進(jìn)行上傳或下載,如果碰到網(wǎng)絡(luò)故障,可以從已經(jīng)上傳或下載的部分開(kāi)始繼續(xù)上傳或者下載未完成的部分,而沒(méi)有必要從頭開(kāi)始上傳或者下載。本文的斷點(diǎn)續(xù)傳主要是針對(duì)斷點(diǎn)上傳場(chǎng)景。

2、應(yīng)用場(chǎng)景

斷點(diǎn)續(xù)傳可以看成是分片上傳的一個(gè)衍生,因此可以使用分片上傳的場(chǎng)景,都可以使用斷點(diǎn)續(xù)傳。

3、實(shí)現(xiàn)斷點(diǎn)續(xù)傳的核心邏輯

在分片上傳的過(guò)程中,如果因?yàn)橄到y(tǒng)崩潰或者網(wǎng)絡(luò)中斷等異常因素導(dǎo)致上傳中斷,這時(shí)候客戶端需要記錄上傳的進(jìn)度。在之后支持再次上傳時(shí),可以繼續(xù)從上次上傳中斷的地方進(jìn)行繼續(xù)上傳。

為了避免客戶端在上傳之后的進(jìn)度數(shù)據(jù)被刪除而導(dǎo)致重新開(kāi)始從頭上傳的問(wèn)題,服務(wù)端也可以提供相應(yīng)的接口便于客戶端對(duì)已經(jīng)上傳的分片數(shù)據(jù)進(jìn)行查詢(xún),從而使客戶端知道已經(jīng)上傳的分片數(shù)據(jù),從而從下一個(gè)分片數(shù)據(jù)開(kāi)始繼續(xù)上傳。

4、實(shí)現(xiàn)流程步驟

a、方案一,常規(guī)步驟

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

b、方案二、本文實(shí)現(xiàn)的步驟

  • 前端(客戶端)需要根據(jù)固定大小對(duì)文件進(jìn)行分片,請(qǐng)求后端(服務(wù)端)時(shí)要帶上分片序號(hào)和大小
  • 服務(wù)端創(chuàng)建conf文件用來(lái)記錄分塊位置,conf文件長(zhǎng)度為總分片數(shù),每上傳一個(gè)分塊即向conf文件中寫(xiě)入一個(gè)127,那么沒(méi)上傳的位置就是默認(rèn)的0,已上傳的就是Byte.MAX_VALUE 127(這步是實(shí)現(xiàn)斷點(diǎn)續(xù)傳和秒傳的核心步驟)
  • 服務(wù)器按照請(qǐng)求數(shù)據(jù)中給的分片序號(hào)和每片分塊大小(分片大小是固定且一樣的)算出開(kāi)始位置,與讀取到的文件片段數(shù)據(jù),寫(xiě)入文件。
5、分片上傳/斷點(diǎn)上傳代碼實(shí)現(xiàn)

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

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

b、后端用兩種方式實(shí)現(xiàn)文件寫(xiě)入,一種是用RandomAccessFile,如果對(duì)RandomAccessFile不熟悉的朋友,可以查看如下鏈接:

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

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

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

后端進(jìn)行寫(xiě)入操作的核心代碼

a、RandomAccessFile實(shí)現(xiàn)方式

@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");
//這個(gè)必須與前端設(shè)定的值一致
longchunkSize=Objects.isNull(param.getChunkSize())?defaultChunkSize*1024*1024
:param.getChunkSize();
longoffset=chunkSize*param.getChunk();
//定位到該分片的偏移量
accessTmpFile.seek(offset);
//寫(xiě)入該分片數(shù)據(jù)
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實(shí)現(xiàn)方式

@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();
//寫(xiě)入該分片數(shù)據(jù)
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、文件操作核心模板類(lèi)代碼

@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();
}

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

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

//completeList檢查是否全部完成,如果數(shù)組里是否全部都是127(全部分片都成功上傳)
byte[]completeList=FileUtils.readFileToByteArray(confFile);
isComplete=Byte.MAX_VALUE;
for(inti=0;i//與運(yùn)算,如果有部分沒(mé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;
}

/**
*把上傳進(jìn)度信息存進(jìn)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保存文件信息到數(shù)據(jù)庫(kù)

}

}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 實(shí)現(xiàn)的后臺(tái)管理系統(tǒng) + 用戶小程序,支持 RBAC 動(dòng)態(tài)權(quán)限、多租戶、數(shù)據(jù)權(quán)限、工作流、三方登錄、支付、短信、商城等功能

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

總結(jié)

在實(shí)現(xiàn)分片上傳的過(guò)程,需要前端和后端配合,比如前后端的上傳塊號(hào)的文件大小,前后端必須得要一致,否則上傳就會(huì)有問(wèn)題。其次文件相關(guān)操作正常都是要搭建一個(gè)文件服務(wù)器的,比如使用fastdfs、hdfs等。

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

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

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

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

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



審核編輯 :李倩


聲明:本文內(nèi)容及配圖由入駐作者撰寫(xiě)或者入駐合作網(wǎng)站授權(quán)轉(zhuǎn)載。文章觀點(diǎn)僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場(chǎng)。文章及其配圖僅供工程師學(xué)習(xí)之用,如有內(nèi)容侵權(quán)或者其他違規(guī)問(wèn)題,請(qǐng)聯(lián)系本站處理。 舉報(bào)投訴
  • 服務(wù)器
    +關(guān)注

    關(guān)注

    12

    文章

    9280

    瀏覽量

    85839
  • spring
    +關(guān)注

    關(guān)注

    0

    文章

    340

    瀏覽量

    14366
  • Boot
    +關(guān)注

    關(guān)注

    0

    文章

    150

    瀏覽量

    35879
  • SpringBoot
    +關(guān)注

    關(guān)注

    0

    文章

    174

    瀏覽量

    189

原文標(biāo)題:SpringBoot超大文件上傳,實(shí)現(xiàn)秒傳

文章出處:【微信號(hào):芋道源碼,微信公眾號(hào):芋道源碼】歡迎添加關(guān)注!文章轉(zhuǎn)載請(qǐng)注明出處。

收藏 人收藏

    評(píng)論

    相關(guān)推薦

    Hadoop 生態(tài)系統(tǒng)在大數(shù)據(jù)處理中的應(yīng)用與實(shí)踐

    基礎(chǔ)。它將大文件分割成多個(gè)數(shù)據(jù)塊,存儲(chǔ)在不同節(jié)點(diǎn)上,實(shí)現(xiàn)高容錯(cuò)性和高擴(kuò)展性。NameNode 負(fù)責(zé)管理文件系統(tǒng)命名空間和元數(shù)據(jù),DataNode 負(fù)責(zé)實(shí)際數(shù)據(jù)存儲(chǔ)。上傳
    的頭像 發(fā)表于 01-21 17:48 ?60次閱讀

    如何使用SFTP傳輸大文件

    在當(dāng)今的數(shù)字化時(shí)代,大文件傳輸變得越來(lái)越常見(jiàn)。無(wú)論是企業(yè)數(shù)據(jù)遷移、遠(yuǎn)程備份還是內(nèi)容分發(fā),都需要一種既安全又高效的文件傳輸方式。SFTP作為一種基于SSH的文件傳輸協(xié)議,提供了一種加密的傳輸方式,確保
    的頭像 發(fā)表于 11-13 14:11 ?923次閱讀

    P2Link是什么?——免費(fèi)讓你體驗(yàn)高效智聯(lián)的新方式

    P2Link 是一個(gè)非常方便、簡(jiǎn)單的工具,可以幫你直接傳輸大文件,或者讓別人訪問(wèn)你內(nèi)網(wǎng)中的服務(wù)。它不像傳統(tǒng)工具需要復(fù)雜的配置,也沒(méi)有上傳到云端的煩惱。特別適合那些臨時(shí)共享文件
    的頭像 發(fā)表于 10-31 14:11 ?249次閱讀

    如何修改buildroot和debian文件系統(tǒng)

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

    如何實(shí)現(xiàn)Python復(fù)制文件操作

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

    請(qǐng)問(wèn)AT支持UDP透嗎?

    做了測(cè)試,SDK V1.00 AT文件燒寫(xiě)后,設(shè)定為T(mén)CP時(shí),透是可以了, 但是設(shè)定為UDP后,透?jìng)鞒霈F(xiàn)發(fā)送不成功,但是接收是可以的,是不是AT不支持UDP透呢? 測(cè)試的指令如下
    發(fā)表于 07-17 08:28

    請(qǐng)問(wèn)SDK固件如何實(shí)現(xiàn)網(wǎng)絡(luò)透

    您好,想請(qǐng)問(wèn)一下,我們現(xiàn)在一個(gè)項(xiàng)目需要用SDK來(lái)實(shí)現(xiàn)網(wǎng)路數(shù)據(jù)透,但是調(diào)用 官方API :espconn_send()會(huì)自動(dòng)把HTTP報(bào)文頭部給添加上去了,沒(méi)有有辦法實(shí)現(xiàn)像AT固件一樣直傳?
    發(fā)表于 07-10 06:59

    兩臺(tái)電腦怎么文件?干貨分享教程

    當(dāng)需要在兩臺(tái)電腦之間傳輸文件時(shí),有多種方便的方法可供選擇,以下是一些常見(jiàn)的方式及教程: ? 使用局域網(wǎng)共享 : 確保兩臺(tái)電腦連接在同一個(gè)局域網(wǎng)內(nèi)。 在其中一臺(tái)電腦上,設(shè)置要共享的文件夾。右鍵點(diǎn)擊
    的頭像 發(fā)表于 07-03 14:39 ?724次閱讀

    esp32如何一次性讀取大文件數(shù)據(jù)?

    esp32沒(méi)有提供數(shù)據(jù)庫(kù)讀寫(xiě)的例子,最近有個(gè)大文件,無(wú)法一次性讀出,請(qǐng)問(wèn),怎么讀取,json中一部分json數(shù)組。然后修改完了以后,在寫(xiě)入進(jìn)去?
    發(fā)表于 06-25 06:52

    使用ESP32-S3開(kāi)發(fā)板http post請(qǐng)求發(fā)送SD卡上的大文件,如何循環(huán)邊讀取文件邊分塊發(fā)送文件呢?

    您和,我準(zhǔn)備使用ESP32-S3開(kāi)發(fā)板http post請(qǐng)求發(fā)送SD卡上的大文件,但是使用esp_http_client_set_post_field的buffer太小,內(nèi)存不能一次性申請(qǐng)?zhí)螅?qǐng)問(wèn)
    發(fā)表于 06-06 06:19

    鴻蒙原生應(yīng)用元服務(wù)開(kāi)發(fā)-Web上傳文件

    Web組件支持前端頁(yè)面選擇文件上傳功能,應(yīng)用開(kāi)發(fā)者可以使用onShowFileSelector()接口來(lái)處理前端頁(yè)面文件上傳的請(qǐng)求。 下面的示例中,當(dāng)用戶在前端頁(yè)面點(diǎn)擊
    發(fā)表于 05-08 11:17

    怎么解決需要在國(guó)外網(wǎng)站上傳文件打開(kāi)和上傳速度慢的問(wèn)題。

    隨著全球化的深入發(fā)展,企業(yè)面臨的挑戰(zhàn)之一是在國(guó)際環(huán)境中保持高效的數(shù)據(jù)傳輸和通信。上傳文件至國(guó)外網(wǎng)站時(shí),傳統(tǒng)的互聯(lián)網(wǎng)連接方式往往無(wú)法滿足企業(yè)對(duì)于速度和穩(wěn)定性的需求,這不僅拖慢了工作進(jìn)度,還可
    的頭像 發(fā)表于 04-28 17:01 ?337次閱讀

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

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

    stm32f303rc usb自定義hid下位機(jī)只要上傳2包數(shù)據(jù),上位機(jī)就無(wú)法上傳了的原因?

    只要下位機(jī)上傳2次數(shù)據(jù),不連續(xù),分開(kāi),哪怕每次數(shù)據(jù)只一個(gè)字節(jié),這時(shí)候,上位機(jī)下傳數(shù)據(jù)立刻失敗,下位機(jī)仍然能夠上傳數(shù)據(jù)。 各位同仁可有指點(diǎn)方向的嗎? 謝謝!
    發(fā)表于 03-29 06:06

    激光焊接機(jī)如何實(shí)現(xiàn)高精度焊接

    編輯:鐳拓激光納激光焊接機(jī)實(shí)現(xiàn)高精度焊接主要依賴(lài)于先進(jìn)的激光技術(shù)和精確的控制系統(tǒng)。以下是鐳拓小編為大家總結(jié)的納激光焊接機(jī)實(shí)現(xiàn)高精度焊接的幾點(diǎn)關(guān)鍵因素:1.激光技術(shù):納
    的頭像 發(fā)表于 01-29 15:38 ?1037次閱讀
    納<b class='flag-5'>秒</b>激光焊接機(jī)如何<b class='flag-5'>實(shí)現(xiàn)</b>高精度焊接
    主站蜘蛛池模板: 999av视频| 东日韩二三区| 国产亚洲高清视频| 少妇高潮惨叫久久久久久欧美| 60岁老年熟妇在线无码| 久久精品中文字幕| 一边捏奶头一边啪高潮会怎么样| 国产亚洲欧洲日韩在线三区| 亚洲国产成人爱AV在线播放丿| 国产亚洲精品网站在线视频 | 双性将军粗壮H灌满怀孕| 超碰97人人做人人爱少妇| 色多多污污下载| 国产精品自产拍在线观看中文 | 亚洲国产在线播放在线| 精品福利一区| 羽月希被黑人吃奶dasd585| 麻豆精品乱码WWW久久密| free乌克兰性xxxxhd| 日本精品在线观看视频| 国产精品久久久久无码AV色戒| 国产叼嘿久久精品久久| 无遮18禁在线永久免费观看挡| 学生妹被爆插到高潮无遮挡| 亚洲欧美一区二区久久| 久久精品中文騷妇女内射| 999国产高清在线精品| 热巴两次用约老师屁股发底线球| 丰满少妇发泄14p| 亚洲精品国产乱码AV在线观看| 久见久热 这里只有精品| 99精品视频免费观看| 色午夜日本高清视频www| 精品久久久久亚洲| asmr淫语| 亚洲国产精品天堂在线播放| 免费夜色污私人影院网站| 国产成人免费网站在线观看| 玉娇龙续集春雪瓶txt免费阅读| 欧美特级特黄a大片免费| 国产亚洲精品久久久久久一区二区|