Вопрос был:
Как вернуть ответ от асинхронного вызова?
, который может быть интерпретирован как:
Как сделать синхронный асинхронный код синхронным?
Решение будет состоять в том, чтобы избежать обратных вызовов и использовать комбинацию Promises и async / await.
Я хотел бы привести пример для запроса Ajax.
(Хотя он может быть записан в Javascript, я предпочитаю писать его на Python и компилировать его в Javascript, используя Transcrypt . Это будет достаточно ясно.)
Позволяет сначала включить использование JQuery, чтобы
$
был доступен какS
:__pragma__ ('alias', 'S', '$')
Определить функцию, которая возвращает Promise, в этом случае вызов Ajax:
def read(url: str): deferred = S.Deferred() S.ajax({'type': "POST", 'url': url, 'data': { }, 'success': lambda d: deferred.resolve(d), 'error': lambda e: deferred.reject(e) }) return deferred.promise()
Использовать асинхронный код, как если бы он был синхронным:
async def readALot(): try: result1 = await read("url_1") result2 = await read("url_2") except Exception: console.warn("Reading a lot failed")
Пока я изучал одни и те же вещи, по существу нет чистого гибридного способа для отслеживания фонового местоположения. Но, если вам интересно, вы можете реализовать то же самое с помощью встроенной реализации iOS.
Прежде чем продолжить отслеживание местоположения Apple Background, вы должны получить несколько подробностей о возможных подходах.
Приложение Apple может отслеживать местоположение двумя способами, и они следующие:
[/g3]
StartUpdatingLocations - это метод инициирования отслеживания местоположения, который вызывает обратный вызов для изменения местоположения, который вызывается каждую секунду до тех пор, пока вы не находитесь на переднем плане / фоновом режиме.
Обратный вызов будет вызываться каждую секунду независимо от того, изменилось ли местоположение или нет, это зависит от метода обработки и принятия решения об изменении местоположения и что на самом деле это изменение местоположения.
StartUpdatingLocations не будет иметь никакого эффекта, если приложение приостановлено / завершено. Это по существу означает, что даже если StartUpdatingLocations вызывается, когда приложение запускается, как только приложение перемещается в фоновом режиме, обратный вызов больше не будет вызываться, даже если есть какое-либо изменение местоположения.
StartUpdatingLocations обеспечивает наиболее точное
Теперь то, что вы пытаетесь сделать, довольно просто в Android, но это не в iOS.
Я бы не стал изобретать цикл, но вы можете изучить полные подробности здесь
. Метод, чтобы постоянно получать местоположение в фоновом режиме для iOS 7 и 8 использует метод «startUpdatingLocation»
[myLocationManager startUpdatingLocation];
, а затем следующий трюк будет на методе делегата «didUpdateLocations». Вам нужно будет использовать таймер и соответствующим образом обработать фоновое задание. Любые отсутствующие шаги и местоположение не будут обновляться непрерывно.
Но в случае получения местоположения, когда приложение завершено / приостановлено, вы не можете использовать [myLocationManager startUpdatingLocation]; Единственный способ заставить его работать: -
[anotherLocationManager startMonitoringSignificantLocationChanges];
Еще один важный трюк - вам нужно будет знать, как обращаться с ключом «UIApplicationLaunchOptionsLocationKey» в делегате приложения «didFinishLaunchingWithOptions». Вот пример кода: -
- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions{
self.shareModel = [LocationShareModel sharedModel];
if ([launchOptions objectForKey:UIApplicationLaunchOptionsLocationKey]) {
self.shareModel.anotherLocationManager = [[CLLocationManager alloc]init];
self.shareModel.anotherLocationManager.delegate = self;
self.shareModel.anotherLocationManager.desiredAccuracy = kCLLocationAccuracyBestForNavigation;
self.shareModel.anotherLocationManager.activityType = CLActivityTypeOtherNavigation;
if(IS_OS_8_OR_LATER) {
[self.shareModel.anotherLocationManager requestAlwaysAuthorization];
}
[self.shareModel.anotherLocationManager startMonitoringSignificantLocationChanges];
}
return YES;
}
В дополнение к методу didFinishLaunchingWithOptions вам нужно создать экземпляр locationManager, когда приложение активно. Вот несколько примеров кода:
- (void)applicationDidEnterBackground:(UIApplication *)application
{
[self.shareModel.anotherLocationManager stopMonitoringSignificantLocationChanges];
if(IS_OS_8_OR_LATER) {
[self.shareModel.anotherLocationManager requestAlwaysAuthorization];
}
[self.shareModel.anotherLocationManager startMonitoringSignificantLocationChanges];
}
- (void)applicationDidBecomeActive:(UIApplication *)application
{
if(self.shareModel.anotherLocationManager)
[self.shareModel.anotherLocationManager stopMonitoringSignificantLocationChanges];
self.shareModel.anotherLocationManager = [[CLLocationManager alloc]init];
self.shareModel.anotherLocationManager.delegate = self;
self.shareModel.anotherLocationManager.desiredAccuracy = kCLLocationAccuracyBestForNavigation;
self.shareModel.anotherLocationManager.activityType = CLActivityTypeOtherNavigation;
if(IS_OS_8_OR_LATER) {
[self.shareModel.anotherLocationManager requestAlwaysAuthorization];
}
[self.shareModel.anotherLocationManager startMonitoringSignificantLocationChanges];
}
Я обнаружил, что мне нужно добавить фоновый режим в имя_проекта-Info.plist в ресурсах в xcode, например здесь:
[/g0]
Теперь он отправляет геолокацию, даже если приложение находится в фоновом режиме или устройство заблокировано.