swift4 - Swift 4 上的几个相互依赖的 JSON 请求
问题描述
我正在重构我之前制作的代码,在那里我使用 Alamofire 下载一些 Json 文件。
第一个请求是直截了当的。我提出请求,得到响应,然后解析它并将其存储在 Realm 中。这里没问题。直截了当的东西。第二个请求有点棘手,因为我需要从第一个 JSON 请求中检索到的几个 ID。
我对这个问题的解决方案是首先在具有 Alamofire 请求的函数上创建一个 Completion 处理程序:
func requestData(httpMethod: String, param: Any?, CallType : String, complition: @escaping (Bool, Any?, Error?) -> Void){
我的想法是使用 Completion 等待 Alamofire 响应完成,然后开始新请求。事实证明这也不起作用。
我能够通过为完成添加延迟来实现这一目标。
DispatchQueue.main.asyncAfter(deadline: .now() + 4)
它确实有效,但由于几个原因远非一个好的实践,我想用更智能的东西来重构它。
我的问题:
1) 在同一个函数上发出许多 JSON 请求的最佳方法是什么?一种正确等待第一个开始第二个的方法,依此类推?2) 现在,我调用一个函数来请求第一个 JSON,并在调用中间发出第二个请求。在我看来,我将第一个请求挂得太久,等待所有请求完成然后完成第一个请求。我认为这不是一个好习惯
这是完整的代码。感谢帮助
@IBAction func getDataButtonPressed(_ sender: Any) {
requestData(httpMethod: "GET", param: nil, CallType: "budgets") { (sucess, response, error) in
if sucess{
print("ready")
DispatchQueue.main.asyncAfter(deadline: .now() + 4){
accounts = realm.objects(Account.self)
requestAccounts()
}
DispatchQueue.main.asyncAfter(deadline: .now() + 4){
users = realm.objects(User.self)
requestUser()
}
}
}
}
func requestData(httpMethod: String, param: Any?, CallType : String, complition: @escaping (Bool, Any?, Error?) -> Void){
let url = "https://XPTO.com/v1/\(CallType)"
guard let urlAddress = URL(string: url) else {return}
var request = URLRequest(url: urlAddress)
request.httpMethod = httpMethod
request.addValue("application/json", forHTTPHeaderField: "accept")
request.addValue("application/json", forHTTPHeaderField: "Content-Type")
request.addValue("Bearer appKey", forHTTPHeaderField: "Authorization")
if param != nil{
guard let httpBody = try? JSONSerialization.data(withJSONObject: param!, options:[]) else {return}
request.httpBody = httpBody
}
Alamofire.request(request).responseJSON { (response) in
let statusCode = response.response?.statusCode
print("Status Code \(statusCode!)")
jsonData = try! JSON(data: response.data!)
complition(true, jsonData, nil)
if httpMethod == "GET"{
saveJsonResponse(jsonData: jsonData, CallType: CallType)
}
}
}
func requestAccounts(){
var count = accounts.count
while count != 0{
let account = accounts[0]
RealmServices.shared.delete(account)
count -= 1
}
let numberOfBugdets = budgets.count
for i in 0...numberOfBugdets - 1{
requestData(httpMethod: "GET", param: nil, CallType: "/budgets/\(budgets[i].id)/accounts") { (sucess, response, error) in
print("accounts downloaded")
let numberOfAccounts = jsonData["data"]["accounts"].count
for j in 0...numberOfAccounts - 1{
let realm = try! Realm()
do{
try realm.write {
// Code to save JSON data to Realm
realm.add(newAccount)
}
} catch {
print("something")
}
}
}
}
}
func requestUser(){
var count = users.count
while count != 0{
let user = users[0]
RealmServices.shared.delete(user)
count -= 1
}
requestData(httpMethod: "GET", param: nil, CallType: "user") { (success, response, error) in
print("User data downloaded")
let realm = try! Realm()
do{
try realm.write {
// Code to save JSON data to Realm
realm.add(newUser)
}
} catch {
print("something")
}
}
}
func saveJsonResponse(jsonData: JSON, CallType: String){
case "budgets":
var count = budgets.count
while count != 0{
let budget = budgets[0]
RealmServices.shared.delete(budget)
count -= 1
}
let numberOfBudgets = jsonData["data"]["budgets"].count
for i in 0...numberOfBudgets - 1 {
// Code to save JSON data to Realm
RealmServices.shared.create(newBudget)
}
}
解决方案
我推荐在这种情况下完成处理程序。
这就是关于如何实现和使用它的代码片段尝试理解它并在代码中实现它的方式。
//CompletionHandlers
func firstOperation(completionHandler: @escaping (_ id: String) -> Void){
//preform alamoFire and in .response { } call completionHandler and pass it the id
completionHandler("10")
}
func buttonClicked () {
firstOperation { (id) in
secondFunction(completionHandler: { (data) in
// your data
})
}
}
func secondFunction(completionHandler: @escaping (_ data: String) -> Void){
//preform alamoFire and in .response { } call completionHandler and pass it the id
completionHandler("some Data")
}
这应该让您更好地理解如何实现它,CompletionHandlers 很强大
特别是在处理这种情况时,当您必须执行取决于其他操作结果的操作时,并且在网络中,我们无论如何都无法预测操作的时间。
在此处阅读有关完成处理程序的更多信息
推荐阅读
- swift - 当 UIView 在滚动 UITableView 时重新出现时动画停止
- sql - 使用分区和大小写求和的 DB2 查询中的条件抛出错误
- php - 使用 OHO Codeigniter 备份 MySQL 数据库
- python - 从 gui 打开一个新的 gui
- webpack - 如何根据环境变量自定义我的 Service Worker?
- excel - 我想修改代码,以便在打开工作簿后立即运行宏
- compilation - x86/x64编译与MSIL的关系
- python - 如何使用 twitter api 从 python 中的特定配置文件打印推文
- javascript - 使用 React 的 Handsontable,每次状态更新时都会重置排序
- python - 从多个熊猫系列中创建一个多维 numpy 元组数组并对其进行处理