swift - Swift - 多个 URL 请求 - 重构和重用的代码
问题描述
我是 Swift 新手,我正在尝试重构我的 URL Post 请求。我在同一个视图控制器中有多个 URL POST 请求,就像这样。一切正常,但在我看来,有很多可以重用的重复代码。特别是,我不知道如何传递/处理应该在 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()
}
然后,您可以根据需要使用适当的参数调用此方法。
推荐阅读
- sql-server - 在sql数据库中最后一次出现“_”后获取子字符串值
- r - 展开 scale_x_continuous()
- dbscan - 考虑到 DBSCAN 算法中所有特征对的组合,为什么考虑所有特征不会复制?
- c++ - 如何在 qt 表单设计器的小部件中显示两个不同的类
- ios - 将 NSData 转换为 NSArray
- c# - 使用 C# 将 Windows 网络配置文件设置为公共
- pandas - 如何使用 pandas 从 csv 文件中读取 3500 行?
- matplotlib - 使用 matplotlib 为组内的条创建带有 xtick 标签的分组条形图
- html - 在 SECTION 内显示 BLOCK
- angular - ngDoCheck VS 使用函数作为属性值 - 性能差异