Swift - запрос нескольких URL - код для рефакторинга и повторного использования - PullRequest
0 голосов
/ 28 февраля 2020

Я новичок в Swift и пытаюсь реорганизовать свои запросы URL Post. У меня есть несколько запросов POST URL внутри одного и того же View Controller, как это. Все работает нормально, но мне кажется, что есть много повторяющегося кода, который можно использовать повторно. В частности, я не знаю, как передавать / обрабатывать разные модели данных, которые должны использоваться в 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-запроса2, который почти идентичен, за исключением параметров и модели, которая будет использоваться для декодирования и действия внутри 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)
        }
    } 

1 Ответ

1 голос
/ 28 февраля 2020

Единственными различиями являются:

  • параметры запроса
  • тип возвращаемой модели
  • действие, которое вы выполняете после получения ответа

Это означает, что мы можем записать это как один единственный метод, принимая три вышеуказанные значения в качестве параметров:

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()
}

Затем вы можете вызвать этот метод с соответствующими параметрами в соответствии с вашими потребностями.

...