在函数外部使用Alamofire请求的值

Vla*_*inu 2 asynchronous ios completionhandler swift alamofire

我似乎无法弄清楚这一点:我试图使用Alamofire请求从我的服务器(在JSON中)获取错误,但我无法获得该函数之外的值.

这是我的功能的实现:

    func alamoRequest(username : String, email: String, password: String, facebook: String, completionHandler : (String?) -> Void) {

    var jsonValue : JSON?
    let URL = "http://someurl.com/login.php"
    let requestParameters = ["username" : username, "email" : email, "password" : password, "facebook" : facebook, "date": NSNull()];
    var jsonString : String = String()

    Alamofire.request(.GET, URL, parameters: requestParameters).validate().responseJSON {
        response in
        switch response.result {
        case .Success:
            if let value = response.result.value {
                jsonValue = JSON(value)
                jsonString = jsonValue!["error"].stringValue
                print("Value in implementation is: \(jsonString)")

            }
        case .Failure(let error):
            print(error)
        }
    }
    completionHandler(jsonString)
}
Run Code Online (Sandbox Code Playgroud)

以下是我称之为:

logic.alamoRequest(usernameField.text!, email: emailField.text!, password: passwordField.text!, facebook: "false") {
            json in
            jsonString = json!
      }

        alamoRequest = jsonString
        print("Value in call is: \(alamoRequest))
Run Code Online (Sandbox Code Playgroud)

但输出总是这样:

Value from call is: Value in implementation is: Email already exists

如何从函数中"返回"某些内容或等待它执行而不阻止UI?

Vic*_*ler 6

你的操作是异步的,等待完成的方式只是使用闭包,你试图在闭包之外返回值,所以还没有检索到值!并且它是空值的原因,你可以completionHandler像在下面的例子中一样"抛出" 你的闭包内部:

func alamoRequest(username : String, email: String, password: String, facebook: String, completionHandler : (String?) -> Void) {

   var jsonValue : JSON?
   let URL = "http://someurl.com/login.php"
   let requestParameters = ["username" : username, "email" : email, "password" : password, "facebook" : facebook, "date": NSNull()];
   var jsonString : String = String()

   Alamofire.request(.GET, URL, parameters: requestParameters).validate().responseJSON {
    response in
       switch response.result {
       case .Success:
           if let value = response.result.value {
               jsonValue = JSON(value)
               jsonString = jsonValue!["error"].stringValue
               print("Value in implementation is: \(jsonString)")
               completionHandler(jsonString)
           }
       case .Failure(let error):
           print(error)
       }
   } 
}
Run Code Online (Sandbox Code Playgroud)

然而,以上述方式,你唯一的出路不知道是否存在着一些错误的请求,因为你completionHandler只调用内部的.Success情况下,如果你愿意,你可以随时调用它,在最后caseenum.

Alamofire.request(.GET, URL, parameters: requestParameters).validate().responseJSON {
    response in
       switch response.result {
       case .Success:
           if let value = response.result.value {
               jsonValue = JSON(value)
               jsonString = jsonValue!["error"].stringValue
               print("Value in implementation is: \(jsonString)")

           }
       case .Failure(let error):
           jsonString = nil
       }
       completionHandler(jsonString)
   } 
}
Run Code Online (Sandbox Code Playgroud)

而如果jsonStringnil一些错误内容时发生,但同样你不知道该错误,任何事情,那么你有两个选择:

  1. 改变你的闭合总是返回jsonStringerror
  2. 将错误封装到throwable闭包中.

第一种情况非常简单,只需更改您的闭包并始终返回错误,如果它nil没有出现错误.

我认为另一种选择更好,如下例所示:

func alamoRequest(username : String, email: String, password: String, facebook: String, 
                  completion: (inner: () throws -> String) -> ()) {

   var jsonValue : JSON?
   let URL = "http://someurl.com/login.php"
   let requestParameters = ["username" : username, "email" : email, "password" : password, "facebook" : facebook, "date": NSNull()];
   var jsonString : String = String()

   Alamofire.request(.GET, URL, parameters: requestParameters).validate().responseJSON {
    response in
       switch response.result {
       case .Success:
           if let value = response.result.value {
               jsonValue = JSON(value)
               jsonString = jsonValue!["error"].stringValue
               print("Value in implementation is: \(jsonString)")
               completionHandler(inner: { return jsonString })
           }
       case .Failure(let error):
           completionHandler(inner: { return error })
       }
   } 
}
Run Code Online (Sandbox Code Playgroud)

然后您可以通过以下方式调用它:

self.alamoRequest(usernameField.text!, email: emailField.text!, password: passwordField.text!, facebook: "false") { (inner: () throws -> String) -> Void in
  do {
    let result = try inner()
  } catch let error {
     print(error)
  }
}
Run Code Online (Sandbox Code Playgroud)

诀窍是该alamoRequest函数采用了一种名为'inner'类型的附加闭包() throws -> String.这个闭包将提供计算结果,或者它将抛出.闭包本身是在计算过程中通过以下两种方法之一构建的:

  • 如果出现错误: inner: {throw error}
  • 如果成功: inner: {return result}

我强烈推荐你一篇关于try/catch在异步调用中使用的优秀文章在Swift中使用try/catch和异步闭包

我希望这对你有帮助.