Как скачать несколько файлов с помощью NSURLSession

У меня проблема. Я пытаюсь сделать несколько загрузок, используя NSURLSession, но я не понял, что я делаю неправильно? Этот класс инициализируется один раз в классе X. startDownload(realm.objects(Music)[indexPath.row]) может вызываться много раз в одном и том же классе. Задачи в классе "Скачать", я это точно знаю. Если вам нужна дополнительная информация, пожалуйста, напишите

class Download: NSObject, NSURLSessionDelegate {
var progress: Float = 0.0
var progressBar: UIProgressView?
var addButton: UIButton?


private var downloadTask: [NSURLSessionDownloadTask] = []
private var backgroundSession: [NSURLSession] = []
private let realm = try! Realm()

private var downloadObject:[Music] = []
private var queueObjects:[Music] = []


func startDownload(object: Music? = nil) {
    if (object != nil) {
        self.queueObjects.append(object!)
    }
    let url = queueObjects[queueObjects.startIndex].url

    if downloadTask.count < 3 {
        let backgroundSessionConfiguration = NSURLSessionConfiguration.backgroundSessionConfigurationWithIdentifier("backgroundSession"+String(queueObjects.count))
        backgroundSession.append(NSURLSession(configuration: backgroundSessionConfiguration, delegate: self, delegateQueue: NSOperationQueue.mainQueue()))
        let sessionIndex = backgroundSession.endIndex-1
        backgroundSession[sessionIndex].sessionDescription = String(sessionIndex)

        downloadTask.append(backgroundSession[sessionIndex].downloadTaskWithURL(NSURL(string: url)!))
        let taskIndex = downloadTask.endIndex-1
        downloadTask[taskIndex].taskDescription = String(taskIndex)
        downloadTask[taskIndex].resume()

        downloadObject.append(queueObjects[queueObjects.startIndex])
        queueObjects.removeAtIndex(queueObjects.startIndex)
    }
}

func URLSession(session: NSURLSession, downloadTask: NSURLSessionDownloadTask, didFinishDownloadingToURL location: NSURL) {

    let index = Int(downloadTask.taskDescription!)!
    print("Index "+String(index))
    let range = downloadObject[ index ].url.rangeOfString("?")!.startIndex.advancedBy(0)
    let url = downloadObject[ index ].url[downloadObject[index].url.startIndex..<range]
    let theFileName = (url as NSString).lastPathComponent
    let path = NSSearchPathForDirectoriesInDomains(NSSearchPathDirectory.DocumentDirectory, NSSearchPathDomainMask.UserDomainMask, true)
    let directoryPath:String = path[0]
    let fileManager = NSFileManager()
    let destinationURLForFile = NSURL(fileURLWithPath: directoryPath.stringByAppendingString( "/"+theFileName))

    if fileManager.fileExistsAtPath(destinationURLForFile.path!){
        print(destinationURLForFile.path!)
        saveObject(downloadObject[index], path: destinationURLForFile.path!)
    }
    else{
        do {
            try fileManager.moveItemAtURL(location, toURL: destinationURLForFile)
            print(destinationURLForFile.path!)
            saveObject(downloadObject[index], path: destinationURLForFile.path!)
        } catch {
            print("An error occurred while moving file to destination url")
        }
    }
    if addButton != nil {
        addButton?.hidden = true
    }
    downloadTask.cancel()
    session.invalidateAndCancel()

    self.backgroundSession[Int(session.sessionDescription!)!].invalidateAndCancel()
    self.backgroundSession.removeAtIndex(Int(session.sessionDescription!)!)
    self.downloadTask[Int(downloadTask.taskDescription!)!].cancel()
    self.downloadTask.removeAtIndex(Int(downloadTask.taskDescription!)!)
}

func URLSession(session: NSURLSession, downloadTask: NSURLSessionDownloadTask, didWriteData bytesWritten: Int64, totalBytesWritten: Int64, totalBytesExpectedToWrite: Int64) {
    progress = Float(totalBytesWritten)/Float(totalBytesExpectedToWrite)
    if progressBar != nil {
        progressBar?.progress = progress
    }
}

private func saveObject(object: Music, path: String) {
    let downloadMusic = DownloadMusic()
    downloadMusic.id = object.id
    downloadMusic.owner_id = object.owner_id
    downloadMusic.artist = object.artist
    downloadMusic.title = object.title
    downloadMusic.duration = object.duration
    downloadMusic.path = path

    try! realm.write() {
        realm.add(downloadMusic)
        downloadObject.removeAtIndex(downloadObject.endIndex-1)
        if self.queueObjects.count > 0 {
            self.startDownload()
        }
        print(queueObjects.count)
        print(downloadObject.count)
        print(downloadMusic)
    }
}

}

Спасибо

1 ответ

Решение

Во-первых, не делай так. Позвольте сеансу ограничить параллелизм для вас. Просто бросьте все запросы сразу.

Во-вторых, не воссоздайте конфигурацию фонового сеанса, если ваше приложение только что не запустилось. Вы должны создать его ровно один раз и больше никогда. Поведение нескольких объектов NSURLSession, указывающих на один и тот же идентификатор, IIRC не определено.

В-третьих, не лишайте законной силы сеанс, пока вы не закончили с ним. Вы отменяете все свои невыполненные запросы, как только закончится первый.

В-четвертых, вы не должны отменять задачи, если вы не хотите остановить текущую задачу. Если задача уже выполнена, ее отмена ничего не дает.

Кроме того, я собираюсь согласиться с людьми, которые сказали, что вам нужно объяснить, что код делает неправильно, прежде чем я смогу помочь в дальнейшем.:-)

Другие вопросы по тегам