0

我正在使用 NextBus API 构建一个公交车预测应用程序,该应用程序将帮助用户获取预测时间和公交车信息。我已经实现了一个函数,它获取用户的当前位置和选择的地址,并返回一个包含 10 条公交路线的列表,这些路线可以最大限度地减少旅行距离和时间。

@IBAction是触发上述功能的:

@IBAction func findAWayPressed(_ sender: UIButton) {
    // Hide confirm button.
    confirmButton.isHidden = true

    // Setup loading HUD.
    let blue = UIColor(red: 153/255, green: 186/255, blue: 221/255, alpha: 1.0)
    SVProgressHUD.setBackgroundColor(blue)
    SVProgressHUD.setStatus("Finding a way for you...")
    SVProgressHUD.setBorderColor(UIColor.black)
    SVProgressHUD.show()

    // Finds a list of ten bus routes that minimizes the distance from the user and their destination.
    WayFinder.shared.findAWay(startCoordinate: origin!, endCoordinate: destination!)

    SVProgressHUD.dismiss()
}

问题是confirmButton.isHidden = trueSVProgressHUD 行似乎只在WayFinder.shared.findAWay()执行后才做任何事情。HUD 显示片刻,然后立即被 关闭SVProgressHUD.dismiss()

这是findAWay功能:

func findAWay(startCoordinate: CLLocationCoordinate2D, endCoordinate: CLLocationCoordinate2D) {
    // Get list of bus routes from NextBus API.
    getRoutes()

    guard !self.routes.isEmpty else {return}

    // Initialize the the lists of destination and origin stops.
    closestDestinations = DistanceData(shortestDistance: 1000000, stops: [])
    closestOrigins = DistanceData(shortestDistance: 1000000, stops: [])

    // Fetch route info for every route in NextBus API.
    var routeConfigsDownloaded: Int = 0
    for route in routes {
        // Counter is always one whether the request fails
        // or succeeds to prevent app crash.
        getRouteInfo(route: route) { (counter) in
            routeConfigsDownloaded += counter
        }
    }

    while routeConfigsDownloaded != routes.count {}

    // Iterate through every stop and retrieve a list
    // of 10 possible destination stops sorted by distance.
    getClosestDestinations(endCoordinate: endCoordinate)
    // Use destination stop routes to find stops near
    // user's current location that end at destination stops.
    getOriginStops(startCoordinate: startCoordinate)

    // Sort routes by adding their orign distance and destination
    // distance and sorting by total distance.
    getFoundWays()
}

private func getRouteInfo(route: Route, completion: @escaping (Int) -> Void) {
    APIWrapper.routeFetcher.fetchRouteInfo(routeTag: route.tag) { (config) in
        if let config = config {
            self.routeConfigs[route.tag] = config
        } else {
            print("Error retrieving route config for Route \(route.tag).")
        }
        completion(1)
    }
}

为什么代码@IBAction不能按顺序执行?findAWay以前叫hud怎么可能不显示在屏幕上?有任何想法吗?

4

1 回答 1

1

因此,您将需要阅读有关“主线程”及其工作原理的信息。也许了解 IOS 主线程

基本上,您是在要求系统显示 HUD,然后执行我认为是长时间运行和阻塞的操作,然后在主线程中全部关闭 HUD。

在方法存在之前,系统不可能显示 HUD,因为它将成为下一个周期的一部分(绘画/布局/其他重要的东西)。在这种情况下,我会倾向于某种“承诺”API,比如PromiseKitHydra,因为它会极大地简化线程的希望。

基本意图是 - 在主线程上显示 HUD,使用后台线程执行查询,完成后关闭 HUD,但在主线程上执行此操作。

这可能看起来像这样..

SVProgressHUD.show()
DispatchQueue.global(qos: .userInitiated).async {
    WayFinder.shared.findAWay(startCoordinate: origin!, endCoordinate: destination!)
    DispatchQueue.main.async {
        SVProgressHUD.dismiss()
    }
}

现在请记住,切勿从主线程上下文之外修改 UI,如果操作系统检测到这一点,它将使您的应用程序崩溃!

我也可以考虑使用 aDispatchSemaphore而不是 "wild running" while-loop,所以而不是..

// Fetch route info for every route in NextBus API.
var routeConfigsDownloaded: Int = 0
for route in routes {
    // Counter is always one whether the request fails
    // or succeeds to prevent app crash.
    getRouteInfo(route: route) { (counter) in
        routeConfigsDownloaded += counter
    }
}

while routeConfigsDownloaded != routes.count {}

您可能会使用类似...

let semaphore = DispatchSemaphore(value: routes.count)
// Fetch route info for every route in NextBus API.
var routeConfigsDownloaded: Int = 0
for route in routes {
    // Counter is always one whether the request fails
    // or succeeds to prevent app crash.
    getRouteInfo(route: route) { (counter) in
        semaphore.signal()
    }
}

semaphore.wait()

这将做同样的事情,但更有效

于 2018-08-01T20:10:37.453 回答