2017-08-14 59 views
1

我试图写在斯威夫特完成处理功能,这是我在斯威夫特代码名为NBObjectiOS的斯威夫特3 CompletionHandler不工作

typealias CompletionHandler = (_ success: Bool, _ data: [String:Any]) -> Void 

// FIND OBJECTS IN BACKGROUND 
func findObjectInBackground(completionHandler: CompletionHandler) { 
    let tbNameStr = "table=" + theTableName 
    var rStr = "" 
    var theData = [String:Any]() 

    for i in 0..<columns.count { 
     rStr += "&c" + "\(i)" + "=" + columns[i] + 
      "&r" + "\(i)" + "=" + records[i] 
    } 
    recordString = tbNameStr + rStr 
    print("RECORD STRING: \(recordString)") 

    let requestURL = URL(string: PATH_TO_API_FOLDER + "queryWhereKeyContains.php?" + recordString) 

    //create the session object 
    let session = URLSession.shared 

    //now create the URLRequest object using the url object 
    let request = URLRequest(url: requestURL!) 

    //create dataTask using the session object to send data to the server 
    let task = session.dataTask(with: request as URLRequest, completionHandler: { data, response, error in 

     //exiting if there is some error 
     if error != nil { 
      print("Error is: \(error!.localizedDescription)") 
      return 
     } else { 

     } 

     do { 
      if let json = try JSONSerialization.jsonObject(with: data!, options: .mutableContainers) as? [String: Any] { 
       theData = json 
       print("JSON: \(json)") 
      } 
     } catch let error { print("ERROR IN PARSING JSON: \(error.localizedDescription)") } 

    }) 

    task.resume() 


    let flag = true 
    completionHandler(flag, theData) 
} 

我调用该函数在我ViewController.swift,像类这样的:

let query = NJObject(tableName: HOTEL_TABLE_NAME) 
query.whereKeyContains(columnName: HOTEL_NAME, record: "Hotel") 
query.findObjectInBackground { (succ, objects) in 
    print("OBJECTS: \(objects)") 
    print("SUCCESS: \(succ)") 
} 

所以,在Xcode的控制台,我正确地让我的JSON数据,但在打印完成处理,data(打印为对象)时是空的。

OBJECTS: [:] 
SUCCESS: true 
JSON: ["objects": <__NSArrayM 0x17004b4c0>(
{ 
    address = "<null>"; 
    createdAt = "2017-08-12 23:08:48"; 
    description = "lorem ipsec dolor sit"; 
    email = "<null>"; 
}, 
{ 
    address = "<null>"; 
    createdAt = "2017-08-14 06:19:10"; 
    description = "lorem ipsec dolor sit"; 
    email = "<null>"; 
}) 
] 

有一两件事我注意到的是,控制台将首先打印对象作为空[:]和成功日志,那么JSON数据。

所以我肯定我的findObjectInBackground()函数有问题,我只是不知道问题出在哪里。

任何帮助将不胜感激!

+1

你是什么意思“它不能正确打印对象”?如果你的意思是'NSArray'部分,那是因为你的字典类型是'[String:Any]',并且当你解析JSON对象时,你不会向下转换为任何显式类型。 –

+0

是的,我的意思是对象:[:]部分,所以我应该在ViewController.swift调用的函数中做什么? – cubycode

+0

所以问题不在于打印,而是完成处理程序为空。看看我的答案。 –

回答

1

你必须把完成处理器数据任务的完成处理

可以省略URLRequest的创作,得到的是默认的,只是通过网址:

let task = session.dataTask(with: requestURL!, completionHandler: { data, response, error in 

    //exiting if there is some error 
    if error != nil { 
     print("Error is: \(error!.localizedDescription)") 
     completionHandler(false, [:]) 
     return 
    } 

    do { 
     if let json = try JSONSerialization.jsonObject(with: data!) as? [String: Any] { 
      print("JSON: \(json)") 
      completionHandler(true, json) // the variable `theData` is not needed. 
     } 
    } catch let error { print("ERROR IN PARSING JSON: \(error.localizedDescription)") } 
     completionHandler(false, [:]) 
}) 

task.resume() 

而且不通过.mutableContainers,这是毫无意义的斯威夫特

边注:在Swift 3这个关闭声明就足够了:

typealias CompletionHandler = (Bool, [String:Any]) -> Void 
+0

它像一个魅力,非常感谢! – cubycode

1

问题是您在网络请求的完成处理程序之外调用完成处理程序,因此您的函数的完成处理程序在异步网络请求返回之前会返回。您需要将其移动到请求的完成处理程序中。

do { 
    if let json = try JSONSerialization.jsonObject(with: data!, options: .mutableContainers) as? [String: Any] { 
     theData = json 
     print("JSON: \(json)") 
     completionHandler(true, theData) 
    } 
}