简体   繁体   中英

parse nested completion handlers

I am trying to use a completion handler in a for loop. The problem is that it will keep running the loop before the completion handler returns since it is an async call. Attached is my code. Do I need to use GCD? I am new (obviously)to swift/ios. Any advice would be much appreciated. Bob

    for srcTerm in sFields {  //search using all search fields

        multiQuery (searchTerm: srcTerm)   {

            if srResult.count < self.lastValue {
                self.lastValue = srResult.count
                self.lastSearch = srcTerm
            }
        }
        // Do more stuff
    }


    func multiQuery (searchTerm: String,  completion: @escaping ([PFObject]) -> ()) {

        var exArray = [PFObject] ()
        let query = PFQuery(className: "searchLinks")

        do {

            query.whereKey("searchTerms", equalTo: searchTerm)

            query.findObjectsInBackground  (block: { (objects, error)-> Void in
                if let error = error {
                    print("Error Generated:  ",error)
                    return
                }
                if let objects = objects {
                   // do stuff
                }
                completion(self.srResult)
            })
        }
    }               // end of function

You could use DispatchGroups, here's an example (taken from https://medium.com/@wilson.balderrama/how-to-use-dispatchgroup-gdc-with-swift-3-35455b9c27e7 . Similar to GCD with nested Parse Queries but updated to Swift 3 API):

// Just a sample function to simulate async calls
func run(after seconds: Int, closure: @escaping () -> Void) {
    let queue = DispatchQueue.global(qos: .background)
    queue.asyncAfter(deadline: .now() + .seconds(seconds)) {
        closure()
    }
}

let group = DispatchGroup()

group.enter()
run(after: 6) {
    print("Hello after 6 seconds")
    group.leave()
}

group.enter()
run(after: 3) {
    print("Hello after 3 seconds")
    group.leave()
}

group.enter()
run(after: 1) {
    print("Hello after 1 second")
    group.leave()
}


group.notify(queue: DispatchQueue.global(qos: .background)) {
    print("All async calls were run!")
}

Using your code:

let group = DispatchGroup()

for srcTerm in sFields {  //search using all search fields
    group.enter()
    multiQuery (searchTerm: srcTerm) {
        if srResult.count < self.lastValue {
            self.lastValue = srResult.count
            self.lastSearch = srcTerm
        }
        group.leave()
    }
}

group.notify(queue: DispatchQueue.global(qos: .background)) {
    // Do something after all async calls are done
}

The technical post webpages of this site follow the CC BY-SA 4.0 protocol. If you need to reprint, please indicate the site URL or the original address.Any question please contact:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM