阿里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>

基础参数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;

    /**
     * 每个线程处理大小 分片大小
     */
    private Integer number;

}

具体上传方法

小文件上传
public static PutObjectResult uploadFile(OssParamDTO ossParamDTO, InputStream inputStream){
    // 创建OSSClient实例。
    OSS ossClient = new OSSClientBuilder().build(ossParamDTO.getEndpoint(), ossParamDTO.getAccessKeyId(), ossParamDTO.getAccessKeySecret());
    PutObjectResult putObjectResult = null;
    // 上传文件流。
    try {
        putObjectResult = ossClient.putObject(ossParamDTO.getBucketName(), ossParamDTO.getObjectName(), inputStream);
        //权限设置
        ossClient.setBucketAcl(ossParamDTO.getBucketName(), CannedAccessControlList.PublicRead);
    } catch (Exception e) {
        e.printStackTrace();
    }finally {
        // 关闭OSSClient。
        ossClient.shutdown();
    }
  return  putObjectResult;
}

大文件上传,分片oss自己处理
处理逻辑:前段轮训查询数据库某个字段,当该字段被回调接口更新时结束轮训,上传完成
public static void uploadBigFile(OssParamDTO ossParamDTO,String path, File file,Long fileId) throws Throwable {
    System.out.println("上传时间:"+System.currentTimeMillis());
    // Endpoint以杭州为例,其它Region请按实际情况填写。
    String endpoint = ossParamDTO.getEndpoint();
    // 阿里云主账号AccessKey拥有所有API的访问权限,风险很高。强烈建议您创建并使用RAM账号进行API访问或日常运维,请登录 https://ram.console.aliyun.com 创建RAM账号。
    String accessKeyId = ossParamDTO.getAccessKeyId();
    String accessKeySecret = ossParamDTO.getAccessKeySecret();

    // 创建OSSClient实例。
    OSS ossClient = new OSSClientBuilder().build(endpoint, accessKeyId, accessKeySecret);

    ObjectMetadata meta = new ObjectMetadata();
    // 指定上传的内容类型。
    meta.setContentType("text/plain");

    // 通过UploadFileRequest设置多个参数。
    UploadFileRequest uploadFileRequest = new UploadFileRequest(ossParamDTO.getBucketName(),ossParamDTO.getObjectName());
    // 指定上传的本地文件。
    uploadFileRequest.setUploadFile(path);
    // 指定上传并发线程数,默认为1。
    uploadFileRequest.setTaskNum(ossParamDTO.getTask());
    // 指定上传的分片大小,范围为100KB~5GB,默认为文件大小/10000。
    uploadFileRequest.setPartSize(ossParamDTO.getNumber() * 1024 * 1024);

    uploadFileRequest.setObjectMetadata(meta);
    // 设置上传成功回调,参数为Callback类型。
    Callback callback = new Callback();
    callback.setCalbackBodyType(Callback.CalbackBodyType.URL);
    //回调参数 --- 同同步到数据库
    callback.setCallbackBody("fileId="+fileId+"&fileName=${object}&uploadStatus=1");
    //回调接口(自己服务器接口,可供外网访问)
    callback.setCallbackUrl("http://3m8wv2.natappfree.cc/web/common/callBack");
    uploadFileRequest.setCallback(callback);

    // 断点续传上传。
    ossClient.uploadFile(uploadFileRequest);
    //权限设置
    ossClient.setBucketAcl(ossParamDTO.getBucketName(), CannedAccessControlList.PublicRead);
    // 关闭OSSClient。
    ossClient.shutdown();
}
大文件本地分片,多线程执行分片上传,再合并碎片
分片上传代码
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);
        // 设置分片大小。除了最后一个分片没有大小限制,其他的分片最小为100KB。
        uploadPartRequest.setPartSize(curPartSize);
        // 设置分片号。每一个上传的分片都有一个分片号,取值范围是1~10000,如果超出这个范围,OSS将返回InvalidArgument的错误码。
        uploadPartRequest.setPartNumber(partNum);
        // 每个分片不需要按顺序上传,甚至可以在不同客户端上传,OSS会按照分片号排序组成完整的文件。
        UploadPartResult uploadPartResult = ossClient.uploadPart(uploadPartRequest);
        // 每次上传分片之后,OSS的返回结果会包含一个PartETag。PartETag将被保存到partETags中。
       log.debug("getPartETag  ::{}" ,uploadPartResult.getPartETag().getETag());
       return uploadPartResult.getPartETag();
    }finally {
        countDownLatch.countDown();
        log.debug("线程: {}  执行完毕, 等待线程数 :{}, 消耗时间: {}",
                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请按实际情况填写。
    String endpoint = ossParamDTO.getEndpoint();
    // 阿里云主账号AccessKey拥有所有API的访问权限,风险很高。强烈建议您创建并使用RAM账号进行API访问或日常运维,请登录 https://ram.console.aliyun.com 创建RAM账号。
    String accessKeyId = ossParamDTO.getAccessKeyId();
    String accessKeySecret = ossParamDTO.getAccessKeySecret();
    String bucketName = ossParamDTO.getBucketName();
    // <yourObjectName>表示上传文件到OSS时需要指定包含文件后缀在内的完整路径,例如abc/efg/123.jpg。
    String objectName = ossParamDTO.getObjectName();

    // 创建OSSClient实例。
    OSS ossClient = new OSSClientBuilder().build(endpoint, accessKeyId, accessKeySecret);

    // 创建InitiateMultipartUploadRequest对象。
    InitiateMultipartUploadRequest request = new InitiateMultipartUploadRequest(bucketName, objectName);

    // 初始化分片。
    InitiateMultipartUploadResult upresult = ossClient.initiateMultipartUpload(request);
    // 返回uploadId,它是分片上传事件的唯一标识,您可以根据这个ID来发起相关的操作,如取消分片上传、查询分片上传等。
    String uploadId = upresult.getUploadId();

    // partETags是PartETag的集合。PartETag由分片的ETag和分片号组成。
    List<PartETag> partETags =  new ArrayList<>();
    // 计算文件有多少个分片 15MB
    final long partSize = 2 * 1024 * 1024L;
    long fileLength = multipartFile.getSize();
    int partCount = (int) (fileLength / partSize);
    if (fileLength % partSize != 0) {
        partCount++;
    }
    // 遍历分片上传。
    log.info("分片数量  {}",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();
        }
        // 跳过已经上传的分片。
        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();
    }
    // 创建CompleteMultipartUploadRequest对象。
    List<PartETag> collect = partETags.stream().sorted(Comparator.comparing(PartETag::getPartNumber)).collect(Collectors.toList());
    // 在执行完成分片上传操作时,需要提供所有有效的partETags。OSS收到提交的partETags后,会逐一验证每个分片的有效性。当所有的数据分片验证通过后,OSS将把这些分片组合成一个完整的文件。
    log.debug("文件开始合并");
    CompleteMultipartUploadRequest completeMultipartUploadRequest =
            new CompleteMultipartUploadRequest(bucketName, objectName, uploadId, collect);

    // 如果需要在完成文件上传的同时设置文件访问权限,请参考以下示例代码。
    completeMultipartUploadRequest.setObjectACL(CannedAccessControlList.PublicRead);
    // 完成上传。
    CompleteMultipartUploadResult completeMultipartUploadResult = ossClient.completeMultipartUpload(completeMultipartUploadRequest);

    // 关闭OSSClient。
    ossClient.shutdown();
    log.debug("消耗总时间:  {}",((System.currentTimeMillis()-before)/1000)+"s");
    return completeMultipartUploadResult;
}

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,335评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,895评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,766评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,918评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,042评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,169评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,219评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,976评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,393评论 1 304
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,711评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,876评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,562评论 4 336
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,193评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,903评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,142评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,699评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,764评论 2 351

推荐阅读更多精彩内容