24

当应用程序被杀死(不是在后台)时,是否可以在 iOS 上使用基于地理的推送通知?

我有兴趣构建一个应用程序,用户将在其中选择地图上的位置,然后如果他/她例如靠近该区域,则会触发基于本地地理的推送通知。

然而,这个“想法”是否可能?GPS 是否可以在应用程序被杀死并运行并在到位时通知用户时运行并比较坐标?是否有关于我可以阅读的主题的任何类型的教程/文章/更多信息?

我在网上阅读的大部分信息更像是实施的一般想法,但没有任何具体的内容。

4

1 回答 1

34

要在应用程序未运行(即先前已终止)时跟踪用户的位置,有两个选项:

  1. 来自“跟踪用户位置”下的iOS 应用程序编程指南:

    对于不需要高精度位置数据的应用,强烈建议使用重大变化位置服务。使用此服务,仅当用户的位置发生重大变化时才会生成位置更新;因此,它非常适合社交应用程序或为用户提供非关键、位置相关信息的应用程序。如果应用程序在发生更新时被挂起,系统会在后台将其唤醒以处理更新。如果应用程序启动此服务然后终止,系统会在新位置可用时自动重新启动应用程序。此服务在 iOS 4 及更高版本中可用,并且仅在包含蜂窝无线电的设备上可用。

    但是,根据CLLocationManager 类参考,它不太准确,更新也不频繁:

    注意:只要设备从之前的通知移动 500 米或更远,应用程序就会收到通知。它不应期望通知的频率超过每五分钟一次。如果设备能够从网络检索数据,则位置管理器更有可能及时传递通知。

  2. 区域监控以类似的方式工作 - 包括在终止后重新启动应用程序 - 但精度更高(取决于 Wifi 网络和手机信号塔的可用性):

    具体的阈值距离由硬件和当前可用的定位技术决定。例如,如果禁用 Wi-Fi,则区域监控的准确性会大大降低。但是,出于测试目的,您可以假设最小距离约为 200 米。

    另一个区域监控考虑是(根据CLLocationManager 类参考)区域进入和退出通知可能仅在跨越区域边界后 3-5 分钟左右收到。

    根据实际需要,区域监控可用于获取“粗略”位置,然后当用户在特定区域内时,在位置管理器上启动更准确的基于GPS的服务。当用户离开感兴趣的区域时,关闭GPS服务以节省电池并再次恢复到粗略的位置监控服务(即区域监控)。这是一个基本的实现:

    SomeViewController.m

    ...
    @interface SomeViewController () <CLLocationManagerDelegate>
    
    @property (nonatomic, strong) CLLocationManager *locationManager;
    @property (nonatomic, strong) CLRegion *someRegion;
    
    @end
    
    @implementation SomeViewController
    
    - (void)viewDidLoad
    {
        [super viewDidLoad];
    
        self.locationManager = [[CLLocationManager alloc] init];
    
        CLLocationDistance radius = 10; // 10 metre sensitivity
        self.someRegion = [[CLRegion alloc] initCircularRegionWithCenter:someCoordinates radius:radius identifier:@"Smithtown Dry Cleaners"];
    
        self.locationManager.delegate = self;
        [self.locationManager startMonitoringForRegion:self.someRegion];
    
        self.locationManager.desiredAccuracy = kCLLocationAccuracyBest;
        self.locationManager.distanceFilter = 10;
        [self.locationManager startUpdatingLocation];
    }
    
    - (void)locationManager:(CLLocationManager *)manager didEnterRegion:(CLRegion *)region
    {
        [self.locationManager startUpdatingLocation];
    }
    
    - (void)locationManager:(CLLocationManager *)manager didExitRegion:(CLRegion *)region
    {
        [self.locationManager stopUpdatingLocation];
    }
    
    // Delegate method from the CLLocationManagerDelegate protocol.
    - (void)locationManager:(CLLocationManager *)manager didUpdateLocations:(NSArray *)locations
    {
        CLLocation* location = [locations lastObject];
    
        // If the user's current location is not within the region anymore, stop updating
        if ([self.someRegion containsCoordinate:location.coordinate] == NO) {
            [self.locationManager stopUpdatingLocation];
        }
    
        NSString *locationData = [NSString stringWithFormat:@"latitude %+.6f, longitude %+.6f\n",
                                  location.coordinate.latitude,
                                  location.coordinate.longitude];
        NSLog(@"%@", locationData);
    
        UILocalNotification *localNotification = [[UILocalNotification alloc] init];
        localNotification.alertBody = locationData;
        localNotification.alertAction = @"Location data received";
        localNotification.hasAction = YES;
        [[UIApplication sharedApplication] presentLocalNotificationNow:localNotification];
    }
    

    请记住将适当的条目添加到应用程序的 plist 文件中,以便应用程序将在后台运行并访问适当的资源:

    MyApp-Info.plist

    <key>UIBackgroundModes</key>
    <array>
            ...
            <string>location</string>
    </array>
    <key>UIRequiredDeviceCapabilities</key>
    <array>
            ...
            <string>location-services</string>
            <string>gps</string>
    </array>
    

    以上代码假设使用iOS6和ARC

于 2013-03-23T22:08:06.200 回答