Swift 3,未调用 URLSession 数据任务完成处理程序

Posted

技术标签:

【中文标题】Swift 3,未调用 URLSession 数据任务完成处理程序【英文标题】:Swift 3, URLSession dataTask completionHandler not called 【发布时间】:2016-08-20 11:27:19 【问题描述】:

我正在编写一个库,所以不使用 UIKit,即使在我的 ios 应用程序中,相同的代码也可以工作,但是当我在命令行中执行时不会。在 PlayGround 中,它似乎也可以工作。

由于某种原因回调没有被触发,所以打印语句没有执行。

internal class func post(request: URLRequest, responseCallback: @escaping (Bool, AnyObject?) -> ()) 
    execTask(request: request, taskCallback:  (status, resp)  -> Void in
            responseCallback(status, resp)
    )


internal class func clientURLRequest(url: URL, path: String, method: RequestMethod.RawValue,  params: Dictionary<String, Any>? = nil) -> URLRequest 
    var request = URLRequest(url: url)
    request.httpMethod = method
    do 
        let jsonData = try JSONSerialization.data(withJSONObject: (params! as [String : Any]), options: .prettyPrinted)

        request.setValue("application/json; charset=utf-8", forHTTPHeaderField: "Content-Type")
        request.httpBody = jsonData
     catch let error as NSError 
        print(error)
    
    return request


private class func execTask(request: URLRequest, taskCallback: @escaping (Bool,
    AnyObject?) -> ()) 

    let session = URLSession(configuration: URLSessionConfiguration.default)
    print("THIS LINE IS PRINTED")
    let task = session.dataTask(with: request, completionHandler: (data, response, error) -> Void in
        if let data = data 
            print("THIS ONE IS NOT PRINTED")
            let json = try? JSONSerialization.jsonObject(with: data, options: [])
            if let response = response as? HTTPURLResponse , 200...299 ~= response.statusCode 
                taskCallback(true, json as AnyObject?)
             else 
                taskCallback(false, json as AnyObject?)
            
        
    )
    task.resume()

Edits -:我正在编写一个库,所以不使用 UIKit,即使在我的 iOS 应用程序中,相同的代码也可以工作,但是当我在命令行中执行时不会。在 PlayGround 中,它似乎也可以工作。

【问题讨论】:

但是如果Both print statements in below code is not executing 那么这与回调无关......只是你的dataTask 方法永远不会被调用。 请告诉我们你如何调用你的dataTask私有方法。 从另一种方法调用它。变量任务正在创建,然后到达 resume()。但completionHandler 没有执行。添加其他有问题的代码。 我现在添加了有问题的整个代码,改变了你建议的方式。还是行不通。仅供参考,我正在使用 xcode-beta-6 "session.dataTask(with: request, completionHandler: (data, response, error) -> Void in" 好的,我会在你建议的重构后一分钟内编辑这篇文章。 【参考方案1】:

我从头开始制作了一个简单的应用程序。 (Xcode 8 beta 6 / swift 3) 在控制器中,我粘贴了您的代码。 (加上网址创建..) 我在调试器中看到了所有内容:

这个是打印出来的

这个也是印刷的

我回来了

所以它似乎工作。

import UIKit

class ViewController: UIViewController 

    override func viewDidLoad() 
        super.viewDidLoad()

        let URLString = "https://apple.com"
        let url = URL(string: URLString)
        let request = URLRequest(url: url!)


        ViewController.execTask(request: request)  (ok, obj) in

            print("I AM BACK")

        

    

    private class func execTask(request: URLRequest, taskCallback: @escaping (Bool,
        AnyObject?) -> ()) 

        let session = URLSession(configuration: URLSessionConfiguration.default)
        print("THIS LINE IS PRINTED")
        let task = session.dataTask(with: request, completionHandler: (data, response, error) -> Void in
            if let data = data 
                print("THIS ONE IS PRINTED, TOO")
                let json = try? JSONSerialization.jsonObject(with: data, options: [])
                if let response = response as? HTTPURLResponse , 200...299 ~= response.statusCode 
                    taskCallback(true, json as AnyObject?)
                 else 
                    taskCallback(false, json as AnyObject?)
                
            
        )
        task.resume()
    


【讨论】:

我正在编写一个命令行实用程序,所以不使用 UIKit,即使在我的应用程序中相同的代码也可以,但是当我在命令行中执行时不会。在 PlayGround 中,它似乎也可以工作。【参考方案2】:

我知道答案迟了,但如果你还没有弄清楚问题或在其他地方遇到问题,让我们试试这个。

您需要将会话变量保存在方法范围之外(使其成为实例变量)。由于您在函数范围内本地定义了它。它在完成处理程序可以被调用之前被释放,记住完成处理程序不能保留你的会话对象,并且在运行循环执行之后,垃圾收集器将释放你的会话对象。每当我们想从委托或完成处理程序中回调时,我们都需要保留这些对象..

self.session = URLSession(configuration: URLSessionConfiguration.default)

【讨论】:

【参考方案3】:

这里建议的更改是否进行了,现在可以使用了。

Using NSURLSession from a Swift command line program

var sema = DispatchSemaphore( value: 0 )

private func execTask(request: URLRequest, taskCallback: @escaping (Bool,
    AnyObject?) -> ()) 

    let session = URLSession(configuration: URLSessionConfiguration.default, delegate: self, delegateQueue: nil )

    session.dataTask(with: request) (data, response, error) -> Void in
        if let data = data 
            let json = try? JSONSerialization.jsonObject(with: data, options: [])
            if let response = response as? HTTPURLResponse , 200...299 ~= response.statusCode 
                taskCallback(true, json as AnyObject?)
             else 
                taskCallback(false, json as AnyObject?)
            
        
    .resume()
    sema.wait()

【讨论】:

taskCallBacks 的解释非常好。【参考方案4】:
let dataTask = session.dataTask(with: request, completionHandler: data, response,error -> Void in 
    print("Request : \(response)")

    let res = response as! HTTPURLResponse

    print("Status Code : \(res.statusCode)")

    let strResponse = NSString(data: data!, encoding: String.Encoding.utf8.rawValue)
    print("Response String :\(strResponse)")
    )
dataTask.resume()

【讨论】:

【参考方案5】:

斯威夫特 3.0

只需将以下代码复制到您的视图控制器中即可。

@IBAction func btnNewApplicationPressed (_ sender: UIButton) 
        callWebService()


func callWebService() 
    // Show MBProgressHUD Here
    var config                              :URLSessionConfiguration!
    var urlSession                          :URLSession!
    
    config = URLSessionConfiguration.default
    urlSession = URLSession(configuration: config)
    
    // MARK:- HeaderField
    let HTTPHeaderField_ContentType         = "Content-Type"
    
    // MARK:- ContentType
    let ContentType_ApplicationJson         = "application/json"
    
    //MARK: HTTPMethod
    let HTTPMethod_Get                      = "GET"
    
    let callURL = URL.init(string: "https://itunes.apple.com/in/rss/newapplications/limit=10/json")
    
    var request = URLRequest.init(url: callURL!)
    
    request.timeoutInterval = 60.0 // TimeoutInterval in Second
    request.cachePolicy = URLRequest.CachePolicy.reloadIgnoringLocalCacheData
    request.addValue(ContentType_ApplicationJson, forHTTPHeaderField: HTTPHeaderField_ContentType)
    request.httpMethod = HTTPMethod_Get
    
    let dataTask = urlSession.dataTask(with: request)  (data,response,error) in
        if error != nil
            return
        
        do 
            let resultJson = try JSONSerialization.jsonObject(with: data!, options: []) as? [String:AnyObject]
            print("Result",resultJson!)
         catch 
            print("Error -> \(error)")
        
    
    
    dataTask.resume()

【讨论】:

考虑解释你的答案。不鼓励仅使用代码的答案。 试过这个答案,没用***.com/questions/42481033/… 首先任务不在主线程中。因此,UI 中的任何内容都需要调用 DispatchQueue.main.async(execute:code),其次您需要添加回调,否则调用将立即返回。这是为 Swift 3 准备的。【参考方案6】:

有时,对我来说,在这些情况下未调用 completionHandler 的解决方案是因为 Info.plist 上的“允许任意加载”标志被定义为 NO。

Allow Arbitrary loads flag defined as YES

【讨论】:

以上是关于Swift 3,未调用 URLSession 数据任务完成处理程序的主要内容,如果未能解决你的问题,请参考以下文章

Swift:UIViewController 实例未从 URLSession 委托方法接收更新

从未调用过 Swift 身份验证挑战 URLSession

从 URLSession 完成处理函数 Swift 3 返回字符串

Swift - 来自URLSession任务的TableView数据源

Swift 3 ObjectMapper - 将 URLSession 结果转换为 Map 对象失败

未调用 URLSession 委托成功方法,但没有错误