Swift 5 URLSession API 类抛出奇怪的错误

问题描述

立即使用 SwiftUI 学习 Swift。我正在尝试构建一个处理 API 的类,但有很多困难。

所以我有一个股票价格函数,我想在有人加载视图时从这个 API 调用它,然后我希望这个函数返回正确的股票价格对象列表,以便我可以将它呈现到图表中。

>

出于某种原因,我遇到了 Result 没有 .success 函数的问题。

很想知道这里的想法!

这是我现在的代码

final class StockAPI{


func fetchData(symbol: String,completion: @escaping (Result) -> Void) {
    let url = URL(string: "https://google.com")!
    
    var request = URLRequest(url: url)
    request.setValue("application/json",forHTTPHeaderField: "Content-Type")
    request.httpMethod = "POST"
    let json: [String: Any] = ["symbol": symbol]
    
    let jsonData = try? JSONSerialization.data(withJSONObject: json)

    request.httpBody = jsonData


    let task = URLSession.shared.dataTask(with: request,completionHandler: { (data,response,error) in
        if let error = error {
            //completion(.error(error.localizedDescription as! Error),error)
            return
      }

        do{
            let stockPrices = try? JSONDecoder().decode([StockPrice].self,from: data!)
            
            dispatchQueue.main.async {
                // update our UI
                print("dispatching request")
                completion(.success(from: stockPrices as! Decoder))
            }
            
        } catch let jsonError {
            return
        }
      
    })
    
    task.resume()
  }

解决方法

在您的两个错误情况(网络错误和解析错误)中,您应该使用 .failure 和错误调用完成闭包。现在你正在盲目飞行。

Result 的使用很奇怪,因为它是一个泛型,通常采用 Result<[StockPrice],Error> 的形式(即指定成功类型是什么,即 [StockPrice],以及失败类型是什么,即Error)。结果是一个泛型,您想指定 .success.failure 的关联类型是什么类型。

所以,它可能是这样的:

func fetchData(symbol: String,completion: @escaping (Result<[StockPrice],Error>) -> Void) { // if you have compilation error here,try `Swift.Result<[StockPrice],Error>` instead
    let url = URL(string: "https://google.com")!

    var request = URLRequest(url: url)
    request.setValue("application/json",forHTTPHeaderField: "Content-Type")
    request.httpMethod = "POST"
    let object = ["symbol": symbol]

    request.httpBody = try! JSONEncoder().encode(object) // or wrap this in `do`-`catch` block and use `try` without the `!`
    let task = URLSession.shared.dataTask(with: request) { data,response,error in
        if let error = error {
            completion(.failure(error))
            return
        }

        do {
            let stockPrices = try JSONDecoder().decode([StockPrice].self,from: data!)

            DispatchQueue.main.async {
                // update our UI
                print("Dispatching request")
                completion(.success(stockPrices))
            }
        } catch let jsonError {
            completion(.failure(jsonError))
        }
    }

    task.resume()
}