/ / Необхідно налаштувати NSJSONSerialization на iOS10 - швидкий, ios10

Потрібно налаштувати NSJSONSerialization на iOS10 - швидкий, ios10

Після оновлення до користувачів iOS10 користувачі почалискаржиться на аварії мого додатка. Я тестую його з iOS10 на тренажері, і дійсно пристрій збігається з повідомленням, що говорить: "Неможливо віддати значення типу" __NSArrayI "в" NSMutableArray "". Ось мій код, будь ласка, допоможіть:

import Foundation

protocol getAllListsModel: class {
func listsDownloadingComplete(downloadedLists: [ContactsList])
}

class ListsDownloader: NSObject, NSURLSessionDataDelegate{

//properties

weak var delegate: getAllListsModel!

var data : NSMutableData = NSMutableData()

func downloadLists() {

let urlPath: String = "http://..."
let url: NSURL = NSURL(string: urlPath)!
var session: NSURLSession!
let configuration =     NSURLSessionConfiguration.ephemeralSessionConfiguration()     //defaultSessionConfiguration()


session = NSURLSession(configuration: configuration, delegate: self, delegateQueue: nil)

let task = session.dataTaskWithURL(url)

task.resume()

}

func URLSession(session: NSURLSession, dataTask: NSURLSessionDataTask, didReceiveData data: NSData) {
self.data.appendData(data);
}

func URLSession(session: NSURLSession, task: NSURLSessionTask, didCompleteWithError error: NSError?) {
if error != nil {
print("Failed to download data")
}else {
self.parseJSON()
print("Lists downloaded")
}

}
func parseJSON() {

var jsonResult: NSMutableArray = NSMutableArray()

do{
try jsonResult =  NSJSONSerialization.JSONObjectWithData(self.data, options:NSJSONReadingOptions.AllowFragments) as! NSMutableArray

} catch let error as NSError {
print(error)
}

var jsonElement: NSDictionary = NSDictionary()
var downloadedLists: [ContactsList] = []

for i in 0...jsonResult.count-1 {

jsonElement = jsonResult[i] as! NSDictionary

let tempContactsList = ContactsList()

//the following insures none of the JsonElement values are nil through optional binding
let id = jsonElement["id"] as? String
let name = jsonElement["name"] as? String
let pin = jsonElement["pin"] as? String
let lastUpdated = jsonElement["created"] as? String
let listAdminDeviceID = jsonElement["admin"] as? String

tempContactsList.id = id
tempContactsList.name = name
tempContactsList.pin = pin
tempContactsList.lastUpdated = lastUpdated
tempContactsList.listAdmin = listAdminDeviceID

downloadedLists.append(tempContactsList)

}

dispatch_async(dispatch_get_main_queue(), { () -> Void in

self.delegate.listsDownloadingComplete(downloadedLists)

})
}
}

Відповіді:

2 для відповіді № 1

Навіть у iOS 9 гарантії не було NSJSONSerialization.JSONObjectWithData(_:options:) повернеться змінний об'єкт чи ні. Ви повинні були вказати NSJSONReadingOptions.MutableContainers.

А у вашому коді ви не модифікуєте jsonResult, що означає, що вам не потрібно оголошувати його як NSMutableArray. Просто замініть NSMutableArray до NSArray, а тоді вам не потрібно вказувати NSJSONReadingOptions.MutableContainers.

Але, як вважає вадіан, краще використовувати Swift типи, а не NSArray або NSDictionary. Цей код повинен працювати як в iOS 9, так і в 10.

func parseJSON() {

var jsonResult: [[String: AnyObject]] = [] //<- use Swift type

do{
try jsonResult =  NSJSONSerialization.JSONObjectWithData(self.data, options: []) as! [[String: AnyObject]] //<- convert to Swift type, no need to specify options

} catch let error as NSError {
print(error)
}

var downloadedLists: [ContactsList] = []

for jsonElement in jsonResult { //<- your for-in usage can be simplified

let tempContactsList = ContactsList()

//the following insures none of the JsonElement values are nil through optional binding
let id = jsonElement["id"] as? String
let name = jsonElement["name"] as? String
let pin = jsonElement["pin"] as? String
let lastUpdated = jsonElement["created"] as? String
let listAdminDeviceID = jsonElement["admin"] as? String

tempContactsList.id = id
tempContactsList.name = name
tempContactsList.pin = pin
tempContactsList.lastUpdated = lastUpdated
tempContactsList.listAdmin = listAdminDeviceID

downloadedLists.append(tempContactsList)

}

dispatch_async(dispatch_get_main_queue(), { () -> Void in

self.delegate.listsDownloadingComplete(downloadedLists)

})
}

Спробуйте це та перевірте його на пристроях iOS 10.

(The as! конверсія призведе до ряду дивних збоїв, коли ваш сервер не працює, але це було б ще однією проблемою, тому я зберігаю його там.)