13

我无法从我的 HTTPrequest 返回数据,也无法让完成处理程序工作。所以请协助我解决这个问题:

public static func createRequest(qMes: message, location: String, method: String) -> String{
    let requestURL = URL(string: location)
    var request = URLRequest(url: requestURL!)

    request.httpMethod = method
    request.httpBody = qMes.toString().data(using: .utf8)

    let requestTask = URLSession.shared.dataTask(with: request) {
        (data: Data?, response: URLResponse?, error: Error?) in

        if(error != nil) {
            print("Error: \(error)")
        }

        return String(data: data!, encoding: String.Encoding.utf8) as String!
    }
    requestTask.resume()
}

它在 void 函数中期望非 void return 语句。到了这一步,我一脸懵逼……

4

4 回答 4

28

您可以使用此完成块方法发送最终响应:

例如: 我在完成块中返回了字符串,在成功响应且没有错误后,只需在块中传递结果。

  public func createRequest(qMes: String, location: String, method: String , completionBlock: @escaping (String) -> Void) -> Void
    {

        let requestURL = URL(string: location)
        var request = URLRequest(url: requestURL!)

        request.httpMethod = method
        request.httpBody = qMes.data(using: .utf8)

        let requestTask = URLSession.shared.dataTask(with: request) {
            (data: Data?, response: URLResponse?, error: Error?) in

            if(error != nil) {
                print("Error: \(error)")
            }else
            {

                let outputStr  = String(data: data!, encoding: String.Encoding.utf8) as String!
                //send this block to required place
                completionBlock(outputStr!);
            }
        }
        requestTask.resume()
    } 

您可以使用以下代码执行上述完成块功能:

 self.createRequest(qMes: "", location: "", method: "") { (output) in

        }

这将解决您的以下要求。

于 2017-03-27T14:03:36.740 回答
5
{
    (data: Data?, response: URLResponse?, error: Error?) in

    if(error != nil) {
        print("Error: \(error)")
    }

    return String(data: data!, encoding: String.Encoding.utf8) as String!
}

这部分代码是该dataTask()方法的完成处理程序。这是您传递dataTask()给稍后执行的方法的代码块(当服务器发回一些数据或出现错误时)。它不会立即执行。

这意味着当您的createRequest()上述方法正在执行时,它会直接传递该代码,然后传递到该requestTask.resume()行,然后该方法结束。此时,因为您的方法被定义为返回 a String,所以您需要返回 a String。从完成处理程序中返回它是不好的,因为它还没有执行,稍后会执行。

处理异步编程有很多不同的方法,但解决这个问题的一种方法是更改​​您的createRequest()方法,使其不被定义为返回 a String,创建一个将 aString作为参数的方法,该方法可以执行任何您想做的操作返回值,然后从完成处理程序中调用该方法。

于 2017-03-27T14:00:09.913 回答
1

就在你的函数调用中

var webString = try String(contentsOf: URL(string: url)!)

你有完整的字符串响应,你可以返回

于 2018-03-02T13:56:53.877 回答
1

如您在问题中提到的,请尝试使用完成处理程序,而不是使用返回。

func createRequest(qMes: message, location: String, method: String, completionHandler: @escaping (_ data:Data?, _ response: URLResponse?, _ error: NSError?) -> Void)

然后return你应该使用类似的东西而不是completionHandler(data, response, error)

这就是您提出请求的方式:

var request = URLRequest(url: Foundation.URL(string: URL)!)
        request.httpMethod = method
        //request.addValue(authString, forHTTPHeaderField: "Authorization") // if you need some

        let task = URLSession.shared.dataTask(with: request, completionHandler: { data, response, error in

            guard error == nil && data != nil else
            {
                print("error=\(error)")
                completionHandler(data, response, error as NSError?)
                return
            }

            completionHandler(data, response, error as NSError?)
        }) 

        task.resume()
于 2017-03-27T14:00:18.697 回答