欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  移动技术

iOS实现文件切片储存并且上传(仿断点续传机制)

程序员文章站 2023-12-17 15:57:22
简介: 我们在开发中,一般在视频类的app或者与硬件交互的app中会有将数据文件上传到云端,少数社交app上传图片也比较多。下面讲的是将数据文件(txt类型)切片储存到本...

简介:

我们在开发中,一般在视频类的app或者与硬件交互的app中会有将数据文件上传到云端,少数社交app上传图片也比较多。下面讲的是将数据文件(txt类型)切片储存到本地并逐片上传到云端模仿断点续传的机制,但事实上,这个操作并不是真正的断点续传。

google了一下,关于切分的文章也蛮多,但是都比较雷同,接下来要分享将数据写进txt并切片储存到本地,一片一片传到云端的整个流程。下面话不多说了,来一起看看详细的介绍吧。

步骤:

       1、将获取到的数据写进文件(总文件)

       2、将总文件按照需求分割并储存到本地

       3、创建数据库(保存分片数据的id、路径等)

       4、根据路径去本地将分片的数据上传到云端

代码:

一、将获取到的数据写进文件

//设置文件路径
nsstring *folder = @"xxx" ;
//创建文件夹
[fileutils createfolderifnotexist:folder];
//获取沙盒路径并且拼接文件路径
nsstring *documentsdirectory= [nshomedirectory() stringbyappendingpathcomponent:@"documents"];
nsstring *filename = [nsstring stringwithformat:@"xxx"];
nsstring *path = [folder stringbyappendingstring:文件名];
nsstring *realpath = [documentsdirectory stringbyappendingstring:path];
//存总数据
nsfilemanager *filemanager = [nsfilemanager defaultmanager];
[filemanager createfileatpath: realpath contents:文件数据 attributes:nil];

二、将总文件按照需求分割并储存到本地

这里你首先要建立数据库表格、设置数据库路径,不熟悉语法可以参考ios数据库fmdb--增删改查(模糊查询)详细介绍

开始分片:这里是用指针的原理

uint32_t point = 0; //移动指针
double offset = 300*1024; //偏移量300kb
double trunkcountfloat = filedatalength/offset; //分片块数
uint32_t trunkcount = trunkcountfloat; //求出浮点型片数

提示:uint32_t为32位无符号类型数据。先初始化一个指针变量,设置偏移量:

1m=1024kb

1kb=1024b

so,1m等于 1024*1024,如果想设置一次切500 kb 就是 500 *1024;

开始分割:原理如下

iOS实现文件切片储存并且上传(仿断点续传机制)有两种情况:

1)可能数据比较小,只切割了一片

if (filedatalength <= offset) { //只有一片
  //插入数据库列表
}

2)切割后大于一片

for(nsuinteger i = 0; i<trunkcount; i++)
  {
  nsdata *trunkdata ;
  if (i == (trunkcount-1)){ //最后一片
    trunkdata = [filedata subdatawithrange:nsmakerange(point, filedatalength -point)];  
  }else{
    trunkdata = [filedata subdatawithrange:nsmakerange(point, offset)]; 
  }
  }

每切一片就将指针向前移动一次

point += offset;

存数据并插入数据库

[filemanager createfileatpath:分片数据路径 contents:分片数据 attributes:nil];
//插入数据库
  ...

如果需要对分割好的数据进行加密,在这一步就可以进行~

三、上传分片数据

这里顺带加上上传的代码:

1)先查询数据库有没有需要上传的代码,有就请求接口上传,每上传成功一片就将数据库对应的分片数据的上传状态码改变,失败就重新请求接口上传,请求的接口一般都需要告诉云端app一共切了多少片数据、片运动数据id、正在上传第几片等。

app可以与云端商量好一种校验方式,比较保证数据安全上传~

上传的代码:(其中一种上传方式)

//拼接数据表单
nsmutableurlrequest *request = [[afhttprequestserializer serializer] multipartformrequestwithmethod:@"post" urlstring:urlstr parameters:dic constructingbodywithblock:^(id<afmultipartformdata > formdata) {   
  [formdata appendpartwithfiledata:txt文件数据 name:字段 filename:txt文件名 mimetype:@"text"];
  } error:&error];  
  if (error) {   
   dlog(@"拼接表单失败");
  }
  //上传
  afurlsessionmanager *manager = [[afurlsessionmanager alloc] initwithsessionconfiguration:[nsurlsessionconfiguration defaultsessionconfiguration]];  manager.operationqueue.maxconcurrentoperationcount = 1;
  manager.responseserializer = [afjsonresponseserializer serializer];
  nsurlsessionuploadtask *uploadtask;
  uploadtask = [manager      uploadtaskwithstreamedrequest:request
      progress:^(nsprogress * _nonnull uploadprogress) {       
      }completionhandler:^(nsurlresponse * _nonnull response, id _nullable responseobject, nserror * _nullable error) {
       //成功的回调
       complete(jsonbackdic); 
       //失败的回调
       fail(error.localizeddescription,error.code);    
      }];
  //最后一定要写上这句代码启动
  [uploadtask resume];

注意:这里最好做一个防止数据重复上传的处理,可以作个判断,初始化一个有序数组nsmutableset,如果该分片数据没有上传就将分片数据的id加进一个有序数组里并执行上传代码,当上传失败就将分片数据的id从有序数组移除并再次发起上传请求。

总结

以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,如果有疑问大家可以留言交流,谢谢大家对的支持。

上一篇:

下一篇: