美文网首页
iOS下载功能的封装

iOS下载功能的封装

作者: Phelthas | 来源:发表于2019-01-12 17:24 被阅读0次

    之前写了 iOS下载功能的实现,但仅仅是把功能实现而已,后来仔细想了想,感觉这个功能跟业务耦合度太高了,还是可以把下载这个功能剥离出来,方便复用。
    抽离出来的代码在这里:https://github.com/Phelthas/LXMDownloader
    这里记录一下思路

    俗话说的好:没有什么解耦是一个中间层搞不定的,如果真的有,那就再加一层!

    抽离封装的主要工作,也就是把能够复用的代码变成一个中间层而已。
    所以首先要确定哪些功能是可以复用的,哪些仅仅是业务代码。
    简单来说,下载这个功能(包括开始,暂停,删除等)是可以复用的,具体下载的是什么,下载完成了要干什么等就是业务范围了。

    1, Model的定义
    我在GitHub上看了好几个star比较多的download库,大部分都是自定义了下载的model,但这个model一旦定了,就已经跟业务挂钩了,因为model肯定跟业务相关,所以如果要封装,就不能自定义具体的model,只能定义下载的对象应该有什么属性之类——这就是协议的应用场景了。
    所以首先就要要定义 下载对象所需要遵守的协议,有了这个协议,就可以说,只要满足这个协议的对象,下载器都可以下载,这就算是个跟业务无关的功能了。

    那model应该有哪些属性呢?

    比较重要的有,url,completedUnitCount,totalUnitCount,downloadStatus,和uniqueId等。
    我看好多别人的库都是直接那url来做唯一标识符,然后将url md5一下作为文件名,但我感觉用url不太科学呀,比如说下载一个视频的时候,视频的videoId是固定的,但视频的url可能会变(为了防盗链,很多视频都有url失效时间),或者一个视频可能有标清,高清等码率,分别对应不同的url,但一般来说下载任何一个码率的视频都算已经下载过该视频了。
    总之我感觉用一个uniqueId来作为下载对象的唯一标识符是很有必要的。
    completedUnitCount和totalUnitCount用来做进度条,一般用kvo或者通知来监听;
    下载速度如果需要的话可以加入一个时间戳,让每次返回的data大小除以时间间隔得出;
    downloadStatus用来表示下载的状态

    public enum LXMDownloaderStatus: Int {
        case none = 0
        case downloading
        case paused
        case waiting
        case finished
        case failed
    }
    

    一个协议定义的属性太多也不方便操作,可以将所有这些属性封装成一个对象,将这个对象作为协议要求的属性。

    @objcMembers
    open class LXMDownloaderItem: NSObject, NSCoding {
    
        //注意:swift的类必须继承NSObject并且明确声明为dynamic才可以使用KVO
    
        open dynamic var downloadStatus: LXMDownloaderStatus = .none
        open dynamic var totalUnitCount: Int64 = 0
        open dynamic var completedUnitCount: Int64 = 0
        open weak var downloadTask: URLSessionDownloadTask? //这里要用weak,让task完成后能正确的结束
        open dynamic var progress: Float {
            if totalUnitCount == 0 {
                return 0
            } else {
                return Float(completedUnitCount) / Float(totalUnitCount)
            }
        }
        open var itemId: String //itemId是唯一标示符,内部使用itemId是否相等来判断是否是同一个对象的
        open var urlString: String
        public init(itemId: String, urlString: String) {
            self.itemId = itemId
            self.urlString = urlString
            super.init()
        }
    
    @objc public protocol LXMDownloaderModelProtocol {
        @objc var lxm_downloadItem: LXMDownloaderItem { set get }
    }
    

    2,Model的持久化
    因为Model肯定是与业务相关的,所以model的持久化也应该由业务方来做,然后在下载器初始化的时候将model赋值给管理器。
    这里因为我这儿定义了对象,所以要为对象加入NSCoding支持。

       /// 注意,encode过程中downloadTask会被忽略,因为URLSessionDownloadTask不能序列化,progress是只读属性,不用序列化
    
        open func encode(with aCoder: NSCoder) {
            aCoder.encode(downloadStatus.rawValue, forKey: "downloadStatus")
            aCoder.encode(totalUnitCount, forKey: "totalUnitCount")
            aCoder.encode(completedUnitCount, forKey: "completedUnitCount")
            aCoder.encode(urlString, forKey: "urlString")
            aCoder.encode(itemId, forKey: "itemId")
        }
    
        public required init?(coder aDecoder: NSCoder) {
            downloadStatus = LXMDownloaderStatus(rawValue: aDecoder.decodeInteger(forKey: "downloadStatus")) ?? .none
            totalUnitCount = aDecoder.decodeInt64(forKey: "totalUnitCount")
            completedUnitCount = aDecoder.decodeInt64(forKey: "completedUnitCount")
            urlString = aDecoder.decodeObject(forKey: "urlString") as? String ?? ""
            itemId = aDecoder.decodeObject(forKey: "itemId") as? String ?? ""
        }
    }
    

    3,下载器

    首先,既然是要抽离封装,那下载器就不应该是一个单例,因为一个APP中可能有不止一个下载器,比如有专门下载视频的videoDownloader,有专门下载文件的fileDownloader,如果是单例就没办法区分了,单例还是应该由业务方来实现。
    这也是参考AFNetworking的思路,AFURLSessionManager和AFHTTPSessionManager虽然都叫Manager,但都不是单例,具体业务在使用的时候,是创建自己的Client作为单例。

    然后下载器的基本功能,包括初始化,开始下载,暂停,取消,删除本地文件等,
    还有工具类的方法,包括返回指定model的本地存放路径,下载文件夹路径,是否存在本地文件的判断等,
    然后下载器的回调,包括下载成功的回调,下载失败的回调,model状态变更需要保存的回调等,
    然后下载器的配置属性,包括最大并发数,是否允许使用蜂窝网络下载。
    这些方法都是针对之前定义好的遵守协议的对象的。

    需要注意的细节:

    1,断点续传的问题
    iOS10开始到 iOS10.2之前的版本resumeData的保存和获取可能会有问题,这个网上也有解决方案,我也就没仔细研究,参考https://benscheirman.com/2016/09/resume-data-broken-in-ios-10/
    https://stackoverflow.com/questions/39346231/resume-nsurlsession-on-ios10/39347461#39347461
    作者给出的代码没有判断iOS10.2之后的版本,我自己测试,这iOS11以后也直接用这段代码的会报错,直接用系统方法就行了。

    2,后台下载的支持

    func application(_ application: UIApplication, handleEventsForBackgroundURLSession identifier: String, completionHandler: @escaping () -> Void)
    

    这个AppDelegate的方法一定要实现!!!
    这个AppDelegate的方法一定要实现!!!
    这个AppDelegate的方法一定要实现!!!

    我看了好几个别人的库,居然没有强调这一点的,这个方法是APP支持后台下载的关键。
    要知道,如果下载的过程中APP进入了后台,那URLSession的所有delegate就都不会在调用了,包括进度的回调,下载完成的回调,下载失败的回调等等,
    如果没有实现上面的方法,而且当下载完成时APP又刚好在后台的话,那下载好的文件是没办法移动到你指定的下载路径的。除非APP又进入了前台,系统才会重新调用URLSession的delegate方法,否则的话下载好的文件就一直在临时文件夹呆着,对用户来说相当于丢失了。
    上面这个代理方法的作用,可以理解为:相当于让APP在用户不知道的情况下进入了前台一次。
    即,只要实现了上面的代理方法,即使APP在后台时,URLSession的- (void)URLSession:(NSURLSession *)session downloadTask:(NSURLSessionDownloadTask *)downloadTask didFinishDownloadingToURL:(NSURL *)location- (void)URLSession:(NSURLSession *)session task:(NSURLSessionTask *)task didCompleteWithError:(NSError *)error方法依然会正常调用。

    3,APP被kill时resumeData的保存问题

    在我看的所有下载库的代码中,就没看到有去处理这个问题的。。。虽然这个问题并不复杂,不知道是大佬们觉得这就根本不是个问题?还是懒的去处理?还是这种情况可以忽略?

    具体场景是:当APP有任务正在下载时被kill掉了(不管是在后台被系统kill掉了还是用户手动kill掉了),那这时候下载到一半的任务其实是有resumeData的。
    但这时候并不会调用到URLSession的didCompleteWithError方法(估计是怕delegate方法还没执行完APP就已经结束了,反而导致未知的问题),这时候系统会发送UIApplication.willTerminateNotification通知,但监听这个通知在回调里执行open func cancel(byProducingResumeData completionHandler: @escaping (Data?) -> Void)也是不行的,因为这个取消方法是异步的,也需要执行时间的,也存在回调还没完成APP就已经结束的问题;

    系统给出的解决方案是:当APP重新启动,同一个identifier的URLSession被创建的时候,这时候执行URLSession的delegate方法didCompleteWithError,其中的error中会带有resumeData,error的类型是 NSURLErrorCancelled,跟正常情况下任务被取消的流程是一致的。

    所以如果想保存APP被kill时的resumeData,应该在APP重新启动的时候在URLSession的delegate方法中保存。
    那么问题来了:APP重新启动回调URLSession的delegate方法时,传入的session和task,跟任务被启动的时候的session和task,不是同一个对象,但他们具有相同的属性值。问题就是如何找到task对应的下载Model,把这个resumeData和对应下载model关联起来?

    上面的model中虽然定义了downloadTask,但downloadTask是不支持序列化的,所以重启以后获取到的downloadTask肯定是nil。回调传入的downloadTask中可以获取的属性有originalRequest,currentRequest,taskDescription,taskIdentifier等,可见,匹配的工作还是需要从这几个属性着手。

    taskDescription应该是最合适的,可以在任务开始的时候给taskDescription赋值,同时将这个taskDescription写入对应下载的model进行持久化,然后回调的时候根据taskDescription来找到对应model。
    (这里还有个问题需要注意一下,AFURLSessionManager已经把taskDescription这个属性给占用了,而且APP重启的时候会给未执行完的task的taskDescription重新赋值)。
    也可以用url来匹配,及利用downloadTask的originalRequest或currentRequest里面的url与下载model里面的url匹配。

    参考:苹果官方教程

    相关文章

      网友评论

          本文标题:iOS下载功能的封装

          本文链接:https://www.haomeiwen.com/subject/cqycdqtx.html