阿里oss文件分片上傳

OSS文件分片上傳

依賴

<!-- https://mvnrepository.com/artifact/com.aliyun.oss/aliyun-sdk-oss -->
<dependency>
    <groupId>com.aliyun.oss</groupId>
    <artifactId>aliyun-sdk-oss</artifactId>
    <version>3.8.1</version>
</dependency>

基礎(chǔ)參數(shù)dto

/**
 * @author WJL
 */
@Data
@Builder
public class OssParamDTO {

    private String endpoint;
    private String accessKeyId;
    private String accessKeySecret;
    private String bucketName;
    private String folder;
    /**
     * objectName = folder + fileName
     */
    private String objectName;

    /**
     * 上傳線程
     */
    private Integer task;

    /**
     * 每個(gè)線程處理大小 分片大小
     */
    private Integer number;

}

具體上傳方法

小文件上傳
public static PutObjectResult uploadFile(OssParamDTO ossParamDTO, InputStream inputStream){
    // 創(chuàng)建OSSClient實(shí)例绎速。
    OSS ossClient = new OSSClientBuilder().build(ossParamDTO.getEndpoint(), ossParamDTO.getAccessKeyId(), ossParamDTO.getAccessKeySecret());
    PutObjectResult putObjectResult = null;
    // 上傳文件流奖蔓。
    try {
        putObjectResult = ossClient.putObject(ossParamDTO.getBucketName(), ossParamDTO.getObjectName(), inputStream);
        //權(quán)限設(shè)置
        ossClient.setBucketAcl(ossParamDTO.getBucketName(), CannedAccessControlList.PublicRead);
    } catch (Exception e) {
        e.printStackTrace();
    }finally {
        // 關(guān)閉OSSClient儒老。
        ossClient.shutdown();
    }
  return  putObjectResult;
}

大文件上傳尺碰,分片oss自己處理
處理邏輯:前段輪訓(xùn)查詢數(shù)據(jù)庫某個(gè)字段德澈,當(dāng)該字段被回調(diào)接口更新時(shí)結(jié)束輪訓(xùn)歇攻,上傳完成
public static void uploadBigFile(OssParamDTO ossParamDTO,String path, File file,Long fileId) throws Throwable {
    System.out.println("上傳時(shí)間:"+System.currentTimeMillis());
    // Endpoint以杭州為例,其它Region請(qǐng)按實(shí)際情況填寫梆造。
    String endpoint = ossParamDTO.getEndpoint();
    // 阿里云主賬號(hào)AccessKey擁有所有API的訪問權(quán)限缴守,風(fēng)險(xiǎn)很高。強(qiáng)烈建議您創(chuàng)建并使用RAM賬號(hào)進(jìn)行API訪問或日常運(yùn)維镇辉,請(qǐng)登錄 https://ram.console.aliyun.com 創(chuàng)建RAM賬號(hào)屡穗。
    String accessKeyId = ossParamDTO.getAccessKeyId();
    String accessKeySecret = ossParamDTO.getAccessKeySecret();

    // 創(chuàng)建OSSClient實(shí)例。
    OSS ossClient = new OSSClientBuilder().build(endpoint, accessKeyId, accessKeySecret);

    ObjectMetadata meta = new ObjectMetadata();
    // 指定上傳的內(nèi)容類型忽肛。
    meta.setContentType("text/plain");

    // 通過UploadFileRequest設(shè)置多個(gè)參數(shù)村砂。
    UploadFileRequest uploadFileRequest = new UploadFileRequest(ossParamDTO.getBucketName(),ossParamDTO.getObjectName());
    // 指定上傳的本地文件。
    uploadFileRequest.setUploadFile(path);
    // 指定上傳并發(fā)線程數(shù)屹逛,默認(rèn)為1础废。
    uploadFileRequest.setTaskNum(ossParamDTO.getTask());
    // 指定上傳的分片大小汛骂,范圍為100KB~5GB,默認(rèn)為文件大小/10000评腺。
    uploadFileRequest.setPartSize(ossParamDTO.getNumber() * 1024 * 1024);

    uploadFileRequest.setObjectMetadata(meta);
    // 設(shè)置上傳成功回調(diào)帘瞭,參數(shù)為Callback類型。
    Callback callback = new Callback();
    callback.setCalbackBodyType(Callback.CalbackBodyType.URL);
    //回調(diào)參數(shù) --- 同同步到數(shù)據(jù)庫
    callback.setCallbackBody("fileId="+fileId+"&fileName=${object}&uploadStatus=1");
    //回調(diào)接口(自己服務(wù)器接口蒿讥,可供外網(wǎng)訪問)
    callback.setCallbackUrl("http://3m8wv2.natappfree.cc/web/common/callBack");
    uploadFileRequest.setCallback(callback);

    // 斷點(diǎn)續(xù)傳上傳蝶念。
    ossClient.uploadFile(uploadFileRequest);
    //權(quán)限設(shè)置
    ossClient.setBucketAcl(ossParamDTO.getBucketName(), CannedAccessControlList.PublicRead);
    // 關(guān)閉OSSClient。
    ossClient.shutdown();
}
大文件本地分片芋绸,多線程執(zhí)行分片上傳媒殉,再合并碎片
分片上傳代碼
PartETag getUploadPartETag(String objectName, String bucketName, String uploadId,
                           InputStream instream, Long curPartSize,Integer partNum,
                           OSS ossClient, CountDownLatch countDownLatch){
    long before = System.currentTimeMillis();
    UploadPartRequest uploadPartRequest = null;
    try {
        log.debug("分片文件上傳線程: {}",Thread.currentThread().getName());
        uploadPartRequest = new UploadPartRequest();
        uploadPartRequest.setBucketName(bucketName);
        uploadPartRequest.setKey(objectName);
        uploadPartRequest.setUploadId(uploadId);
        uploadPartRequest.setInputStream(instream);
        // 設(shè)置分片大小。除了最后一個(gè)分片沒有大小限制侥钳,其他的分片最小為100KB适袜。
        uploadPartRequest.setPartSize(curPartSize);
        // 設(shè)置分片號(hào)。每一個(gè)上傳的分片都有一個(gè)分片號(hào)舷夺,取值范圍是1~10000苦酱,如果超出這個(gè)范圍,OSS將返回InvalidArgument的錯(cuò)誤碼给猾。
        uploadPartRequest.setPartNumber(partNum);
        // 每個(gè)分片不需要按順序上傳疫萤,甚至可以在不同客戶端上傳,OSS會(huì)按照分片號(hào)排序組成完整的文件敢伸。
        UploadPartResult uploadPartResult = ossClient.uploadPart(uploadPartRequest);
        // 每次上傳分片之后扯饶,OSS的返回結(jié)果會(huì)包含一個(gè)PartETag。PartETag將被保存到partETags中池颈。
       log.debug("getPartETag  ::{}" ,uploadPartResult.getPartETag().getETag());
       return uploadPartResult.getPartETag();
    }finally {
        countDownLatch.countDown();
        log.debug("線程: {}  執(zhí)行完畢, 等待線程數(shù) :{}, 消耗時(shí)間: {}",
                Thread.currentThread().getName(),countDownLatch.getCount(),
                ((System.currentTimeMillis()-before)/1000)+"s");
    }
}

外部分片代碼
@Qualifier("taskExecutor")
@Autowired
ThreadPoolTaskExecutor taskExecutor;
/**
* 上傳
* @param ossParamDTO
* @param multipartFile
* @return
*/
public CompleteMultipartUploadResult uploadBigFileForProd(OssParamDTO ossParamDTO, MultipartFile multipartFile){
    Long before = System.currentTimeMillis();
    // Endpoint以杭州為例尾序,其它Region請(qǐng)按實(shí)際情況填寫。
    String endpoint = ossParamDTO.getEndpoint();
    // 阿里云主賬號(hào)AccessKey擁有所有API的訪問權(quán)限躯砰,風(fēng)險(xiǎn)很高每币。強(qiáng)烈建議您創(chuàng)建并使用RAM賬號(hào)進(jìn)行API訪問或日常運(yùn)維,請(qǐng)登錄 https://ram.console.aliyun.com 創(chuàng)建RAM賬號(hào)琢歇。
    String accessKeyId = ossParamDTO.getAccessKeyId();
    String accessKeySecret = ossParamDTO.getAccessKeySecret();
    String bucketName = ossParamDTO.getBucketName();
    // <yourObjectName>表示上傳文件到OSS時(shí)需要指定包含文件后綴在內(nèi)的完整路徑兰怠,例如abc/efg/123.jpg。
    String objectName = ossParamDTO.getObjectName();

    // 創(chuàng)建OSSClient實(shí)例李茫。
    OSS ossClient = new OSSClientBuilder().build(endpoint, accessKeyId, accessKeySecret);

    // 創(chuàng)建InitiateMultipartUploadRequest對(duì)象揭保。
    InitiateMultipartUploadRequest request = new InitiateMultipartUploadRequest(bucketName, objectName);

    // 初始化分片。
    InitiateMultipartUploadResult upresult = ossClient.initiateMultipartUpload(request);
    // 返回uploadId魄宏,它是分片上傳事件的唯一標(biāo)識(shí)秸侣,您可以根據(jù)這個(gè)ID來發(fā)起相關(guān)的操作,如取消分片上傳、查詢分片上傳等味榛。
    String uploadId = upresult.getUploadId();

    // partETags是PartETag的集合方篮。PartETag由分片的ETag和分片號(hào)組成。
    List<PartETag> partETags =  new ArrayList<>();
    // 計(jì)算文件有多少個(gè)分片 15MB
    final long partSize = 2 * 1024 * 1024L;
    long fileLength = multipartFile.getSize();
    int partCount = (int) (fileLength / partSize);
    if (fileLength % partSize != 0) {
        partCount++;
    }
    // 遍歷分片上傳励负。
    log.info("分片數(shù)量  {}",partCount);
    List<Future<PartETag>> futureList = Collections.synchronizedList(new ArrayList());
    CountDownLatch countDownLatch = new CountDownLatch(partCount);
    for (int i = 0; i < partCount; i++) {
        long startPos = i * partSize;
        long curPartSize = (i + 1 == partCount) ? (fileLength - startPos) : partSize;
        InputStream instream = null;
        try {
            instream = multipartFile.getInputStream();
        }  catch (IOException e) {
            e.printStackTrace();
        }
        // 跳過已經(jīng)上傳的分片。
        try {
            instream.skip(startPos);
        } catch (IOException e) {
            e.printStackTrace();
        }
        int finalI = i;
        InputStream finalInstream = instream;
        Future<PartETag> partETagFuture = taskExecutor.submit(() ->
                fileServiceExtAsync.getUploadPartETag(objectName, bucketName, uploadId, finalInstream, curPartSize, finalI + 1, ossClient, countDownLatch));
        futureList.add(partETagFuture);
    }
    try {
        countDownLatch.await();
        for (Future<PartETag> tagFuture : futureList) {
            partETags.add(tagFuture.get());
        }
    } catch (InterruptedException | ExecutionException e) {
        e.printStackTrace();
    }
    // 創(chuàng)建CompleteMultipartUploadRequest對(duì)象匕得。
    List<PartETag> collect = partETags.stream().sorted(Comparator.comparing(PartETag::getPartNumber)).collect(Collectors.toList());
    // 在執(zhí)行完成分片上傳操作時(shí)继榆,需要提供所有有效的partETags。OSS收到提交的partETags后汁掠,會(huì)逐一驗(yàn)證每個(gè)分片的有效性略吨。當(dāng)所有的數(shù)據(jù)分片驗(yàn)證通過后,OSS將把這些分片組合成一個(gè)完整的文件考阱。
    log.debug("文件開始合并");
    CompleteMultipartUploadRequest completeMultipartUploadRequest =
            new CompleteMultipartUploadRequest(bucketName, objectName, uploadId, collect);

    // 如果需要在完成文件上傳的同時(shí)設(shè)置文件訪問權(quán)限翠忠,請(qǐng)參考以下示例代碼。
    completeMultipartUploadRequest.setObjectACL(CannedAccessControlList.PublicRead);
    // 完成上傳乞榨。
    CompleteMultipartUploadResult completeMultipartUploadResult = ossClient.completeMultipartUpload(completeMultipartUploadRequest);

    // 關(guān)閉OSSClient秽之。
    ossClient.shutdown();
    log.debug("消耗總時(shí)間:  {}",((System.currentTimeMillis()-before)/1000)+"s");
    return completeMultipartUploadResult;
}

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市吃既,隨后出現(xiàn)的幾起案子考榨,更是在濱河造成了極大的恐慌,老刑警劉巖鹦倚,帶你破解...
    沈念sama閱讀 218,525評(píng)論 6 507
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件河质,死亡現(xiàn)場離奇詭異,居然都是意外死亡震叙,警方通過查閱死者的電腦和手機(jī)掀鹅,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,203評(píng)論 3 395
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來媒楼,“玉大人乐尊,你說我怎么就攤上這事∠蛔” “怎么了科吭?”我有些...
    開封第一講書人閱讀 164,862評(píng)論 0 354
  • 文/不壞的土叔 我叫張陵,是天一觀的道長猴鲫。 經(jīng)常有香客問我对人,道長,這世上最難降的妖魔是什么拂共? 我笑而不...
    開封第一講書人閱讀 58,728評(píng)論 1 294
  • 正文 為了忘掉前任牺弄,我火速辦了婚禮,結(jié)果婚禮上宜狐,老公的妹妹穿的比我還像新娘势告。我一直安慰自己蛇捌,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,743評(píng)論 6 392
  • 文/花漫 我一把揭開白布咱台。 她就那樣靜靜地躺著络拌,像睡著了一般。 火紅的嫁衣襯著肌膚如雪回溺。 梳的紋絲不亂的頭發(fā)上春贸,一...
    開封第一講書人閱讀 51,590評(píng)論 1 305
  • 那天,我揣著相機(jī)與錄音遗遵,去河邊找鬼萍恕。 笑死,一個(gè)胖子當(dāng)著我的面吹牛车要,可吹牛的內(nèi)容都是我干的允粤。 我是一名探鬼主播,決...
    沈念sama閱讀 40,330評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼翼岁,長吁一口氣:“原來是場噩夢(mèng)啊……” “哼类垫!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起琅坡,我...
    開封第一講書人閱讀 39,244評(píng)論 0 276
  • 序言:老撾萬榮一對(duì)情侶失蹤阔挠,失蹤者是張志新(化名)和其女友劉穎,沒想到半個(gè)月后脑蠕,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體购撼,經(jīng)...
    沈念sama閱讀 45,693評(píng)論 1 314
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,885評(píng)論 3 336
  • 正文 我和宋清朗相戀三年谴仙,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了迂求。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 40,001評(píng)論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡晃跺,死狀恐怖揩局,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情掀虎,我是刑警寧澤凌盯,帶...
    沈念sama閱讀 35,723評(píng)論 5 346
  • 正文 年R本政府宣布,位于F島的核電站烹玉,受9級(jí)特大地震影響驰怎,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜二打,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,343評(píng)論 3 330
  • 文/蒙蒙 一县忌、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦症杏、人聲如沸装获。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,919評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽穴豫。三九已至,卻和暖如春逼友,著一層夾襖步出監(jiān)牢的瞬間绩郎,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 33,042評(píng)論 1 270
  • 我被黑心中介騙來泰國打工翁逞, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人溉仑。 一個(gè)月前我還...
    沈念sama閱讀 48,191評(píng)論 3 370
  • 正文 我出身青樓挖函,卻偏偏與公主長得像,于是被迫代替她去往敵國和親浊竟。 傳聞我的和親對(duì)象是個(gè)殘疾皇子怨喘,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,955評(píng)論 2 355

推薦閱讀更多精彩內(nèi)容