Swift - 多个 URL 请求 - 代码重构和重用

Swift - Multiple URL Request - Code To Refactor and To Reuse

我是 Swift 的新手,我正在尝试重构我的 URL Post 请求。我在同一个视图控制器中有多个 URL POST 请求,就像这样。一切正常,但在我看来,有很多重复的代码可以重用。特别是,我不知道如何 pass/handle 应该在 parseRequest1 和 parseRequest2 中使用不同的数据模型。我还读到在同一个项目中应该只有一个会话用于 URL 请求。任何帮助将不胜感激!

func request1() {        
    let parameters = [...//some parameters to send]
    guard let url = URL(string: "https//www.....") else {return}
    var request = URLRequest(url: url)
    request.httpMethod = "POST"
    request.addValue("application/json", forHTTPHeaderField: "Content-Type")
    guard let parametersToSend = try? JSONSerialization.data(withJSONObject: parameters, options: []) 
       else {
         print("Error")
         return
       }
    request.httpBody = parametersToSend
    let session = URLSession.shared
    session.dataTask(with: request) { (data, response, error) in
       if let safeData = data {
          self.parseRequest1(data: safeData)
        }
   }.resume()
}

func parseRequest1(data: Data){
    let decoder = JSONDecoder()
    do{
      let decodedData = try decoder.decode(DataModelForRequest1.self, from: data)
        DispatchQueue.main.async {
           self.performAction1(request1Result)
        }
    } catch {
       print(error)
    }
} 

然后我有另一个 URL 请求 request2 除了参数和用于在 parseRequest2 中解码和操作的模型外几乎相同。

func request2() {        
        let parameters = [...//some parameters to send]
        guard let url = URL(string: "https//www.....") else {return}
        var request = URLRequest(url: url)
        request.httpMethod = "POST"
        request.addValue("application/json", forHTTPHeaderField: "Content-Type")
        guard let parametersToSend = try? JSONSerialization.data(withJSONObject: parameters, options: []) 
           else {
             print("Error")
             return
           }
        request.httpBody = parametersToSend
        let session = URLSession.shared
        session.dataTask(with: request) { (data, response, error) in
           if let safeData = data {
              self.parseRequest2(data: safeData)
            }
       }.resume()
    }

    func parseRequest2(data: Data){
        let decoder = JSONDecoder()
        do{
          let decodedData = try decoder.decode(DataModelForRequest2.self, from: data)
            DispatchQueue.main.async {
               self.performAction2(request2Result)
            }
        } catch {
           print(error)
        }
    } 

唯一的区别似乎是:

  • 请求参数
  • 返回的模型类型
  • 您在收到回复后执行的操作

这意味着我们可以将其写成一个单一的方法,将上述三个值作为参数:

func request<T: Codable>(modelType: T.Type, parameters: [String: Any], completion: (T) -> Void) {

    func parseResponse(data: Data){
        let decoder = JSONDecoder()
        do{
          let decodedData = try decoder.decode(T.self, from: data)
            DispatchQueue.main.async {
              completion(decodedData)
            }
        } catch {
           print(error)
        }
    } 

    guard let url = URL(string: "https//www.....") else {return}
    var request = URLRequest(url: url)
    request.httpMethod = "POST"
    request.addValue("application/json", forHTTPHeaderField: "Content-Type")
    guard let parametersToSend = try? JSONSerialization.data(withJSONObject: parameters, options: []) 
       else {
         print("Error")
         return
       }
    request.httpBody = parametersToSend
    let session = URLSession.shared
    session.dataTask(with: request) { (data, response, error) in
       if let safeData = data {
          parseResponse(data: safeData)
        }
   }.resume()
}

然后您可以根据需要使用适当的参数调用此方法。