Getting the User’s Location? ios官方文檔:https://developer.apple.com/library/ios/documentation/UserExperience/Conceptual/LocationAwarenessPG/CoreLocation/CoreLocation.html#//apple_ref/doc/uid/TP40009497-CH2-SW8
ios想要獲取用戶位置信息腾窝,就需要開啟定位服務荠藤。開啟定位之前糯俗,工程中需要導入蘋果自帶的#import CoreLocation/CoreLocation.h 框架。
定位有3種模式:基站定位,GPS定位厌蔽,WiFi定位: 在工程Info.plist中加入UIRequiredDeviceCapabilities鍵 指定為數(shù)組 其中加入location-services 字符串或gps字符串 來指定定位的模式,不過也不用加入,一般定位都會默認了這幾種方式桥温,官方說明 If your iOS app uses location services but is able to operate successfully without them, don’t include the corresponding strings in theUIRequiredDeviceCapabilitieskey,意思就是 不需要你來加入了梁丘。這里只是提示一下侵浸。
開啟定位:一定要在工程Info.plist加入NSLocationAlwaysUsageDescription/NSLocationWhenInUseUsageDescription鍵 獲取定位權限,說明獲取定位的描述氛谜,這個會在你開啟定位的時候系統(tǒng)自動彈出一個是否開啟的提示框掏觉,提示說明定位描述。
首先得創(chuàng)建一個定位的實例
- (CLLocationManager*)locationManager{
if(_locationManager==nil) {
_locationManager= [[CLLocationManageralloc]init];
_locationManager.desiredAccuracy=kCLLocationAccuracyNearestTenMeters;//定位精度
_locationManager.distanceFilter=kCLLocationAccuracyHundredMeters;//定位更新距離
//_locationManager.allowsBackgroundLocationUpdates=YES;//UIBackgroundModes 設置了 "location" 一定要設置此為Yes
//_locationManager.pausesLocationUpdatesAutomatically=YES;//UIBackgroundModes 設置了 "location"值漫,為了省電提供的一種方式澳腹,yes 容許定位自動暫停,
_locationManager.delegate=self;
}
return_locationManager;
}
- (void)startUpdateLocation
{
// [CLLocationManager authorizationStatus] == kCLAuthorizationStatusDenied
//檢查定位是否開啟
// iOS8以后需要申請地理搜索權限
if([CLLocationManagerlocationServicesEnabled]) {
if([self.locationManagerrespondsToSelector:@selector(requestAlwaysAuthorization)]) {
[self.locationManager ?requestAlwaysAuthorization];
}else{
// 8.0以下版本杨何,直接搜索地理位置
[self.locationManager ?startUpdatingLocation];
}
}else{
UIAlertView*alter = [[UIAlertViewalloc]initWithTitle:@"提示"message:@"手機定位功能不正常酱塔,請檢查手機是否正常"delegate:nilcancelButtonTitle:@"確定"otherButtonTitles:nil];
[altershow];
//[self.locationManager startUpdatingLocation];
}
}
//只要CLAuthorizationStatus 定位的狀態(tài)發(fā)生改變就會調(diào)用這個方法 你在使用 app的過程中切換定位狀態(tài)也還是會 調(diào)用這個方法
- (void)locationManager:(CLLocationManager*)manager didChangeAuthorizationStatus:(CLAuthorizationStatus)status
{//significantLocationChangeMonitoringAvailable
if(status ==kCLAuthorizationStatusAuthorizedAlways) {
// iOS8.0以后,只有獲取了權限危虱,才能開始搜索地理位置
self.isLocationServiceOn=YES;
[self.locationManager startUpdatingLocation];
}elseif(status ==kCLAuthorizationStatusDenied){
//UIAlertController *alert = [UIAlertController alertControllerWithTitle:@"提示" message:@"定位沒有開啟Q蛲蕖!埃跷!\n請到設置->隱私->位置\n中開啟定位服務"preferredStyle:UIAlertControllerStyleAlert];
//UIAlertAction *action1 = [UIAlertAction actionWithTitle:@"知道了" style:UIAlertActionStyleDefault handler:nil];
//[alert addAction:action1];
//[[[UIApplication sharedApplication].windows lastObject].rootViewController presentViewController:alert animated:YES completion:nil];
self.isLocationServiceOn=NO;
}else{
}
}
- (void)locationManager:(CLLocationManager*)manager
didUpdateLocations:(NSArray*)locations
{
//self.myHud.labelText = @"正在定位地理位置...";
//[self.myHud show:YES];
AppMgr*mgr = [AppMgrdefaultAppMgr];
CLLocation* loc = [locationslastObject];
mgr.originCoord= loc.coordinate;
//這里蕊玷,查詢百度地圖的經(jīng)緯度
NSDictionary* dict =BMKConvertBaiduCoorFrom(mgr.originCoord,BMK_COORDTYPE_GPS);
floatlatitude = [[mgrdecodeBase64FromString:[dictvalueForKey:@"y"]]floatValue];
floatlongitude = [[mgrdecodeBase64FromString:[dictvalueForKey:@"x"]]doubleValue];
mgr.baiduCoord=CLLocationCoordinate2DMake(latitude, longitude);
// mgr.baiduCoord = mgr.originCoord;
NSLog(@"origincoord latitude:%f, longitude:%f", mgr.originCoord.latitude, mgr.originCoord.longitude);
TLog(@"轉(zhuǎn)換百度坐標:%lf,%lf",mgr.baiduCoord.longitude,mgr.baiduCoord.latitude);
//業(yè)務需要拿到定位城市邮利,根據(jù)定位城市來下載項目數(shù)據(jù)包,為了保證一定能拿到城市所以這里需要在登陸的時候僅加載一次垃帅,以后的定位調(diào)用不能執(zhí)行
if(!self.onlyDoOnce&&locations.count>0) {
self.onlyDoOnce=YES;
CLGeocoder*geocoder = [[CLGeocoderalloc]init];
[geocoderreverseGeocodeLocation:loccompletionHandler:^(NSArray*placemarks,NSError*error) {
if(error) {
//反地理編碼錯誤
NSLog(@"反地理編碼錯誤:%@",error);
mgr.currentCity= [[fmCityObjalloc]init];
}else{
//編碼正確拿到城市開始加載城市的資源包
if(placemarks.count>0) {
CLPlacemark* mark = [placemarksfirstObject];
NSString*cityName = mark.locality;
if(!cityName) {
//四大直轄市的城市信息無法通過locality獲得延届,只能通過獲取省份的方法來獲得(如果city為空,則可知為直轄市)
cityName = mark.administrativeArea;
}
NSString* city = [selfgetPriciseCityName:cityNamestateFlag:mark.administrativeArea];
NSLog(@"定位城市:%@", city);
//在這里確定當前城市的信息挺智,名稱祷愉,代碼號
[selfcheckLocalState:city];
}
}
}];
}
}
- (void)locationManager:(CLLocationManager*)manager didFailWithError:(NSError*)error
{
//定位失敗了的情況且只在登陸的時候執(zhí)行一次
if(error) {
self.isLocationServiceOn=NO;
NSLog(@"手機定位失敗");
}
}
蘋果官方給出了3中定位方式:并說明為了節(jié)省電池電量 需要開發(fā)者選擇合適的定位方式,特別是后臺模式的定位赦颇,開啟定位(NSLocationAlwaysUsageDescription/NSLocationWhenInUseUsageDescription)一般是不需要設定后臺模式的二鳄,除非你app切換到后臺后還需要傳定位數(shù)據(jù),如下圖你開啟了后臺定位模式
Performance - 2.5.4? ? Your app declares support for location in the UIBackgroundModes key in your Info.plist file but does not have any features that require persistent location. Apps that declare support for location in the UIBackgroundModes key in your Info.plist file must have features that require persistent location.
蘋果審核的時候媒怯,必須要說明你哪里用到了定位订讼,且是用來干什么的(拍成視屏給蘋果說明)才能通過審核,不然就會被拒扇苞。
1.標準的定位服務欺殿,可以獲取到用戶當前的定位位置信息.
? ? 上面的代碼就是標準的定位服務, 如果你是開啟了后臺定位模式即UIBackgroundModes 設置了 "location"/一定要設置_locationManager.allowsBackgroundLocationUpdates=YES;_locationManager.pausesLocationUpdatesAutomatically=YES鳖敷,提交蘋果審核時要說明后臺模式的用處脖苏。
2.區(qū)域監(jiān)測的定位方式,通過監(jiān)測已經(jīng)指定一些地理區(qū)域定踱,來返回當前用戶定位的地理位置信息棍潘。或則崖媚,檢 ? ?測藍牙區(qū)域亦歉,如果檢測到當前位置有藍牙服務,系統(tǒng)會返回用戶當前定位的位置畅哑。
? ? ? 這個一般是用在藍牙方面肴楷,可以得到地理位置,地理定位都是用的其他2種定位方式?
3. 明顯的位置變化定位方式荠呐,離上次位置赛蔫,用戶位置有一定的距離變化 例如 500 meters or more,就會返回定位位置信息泥张。即使位置沒有變化呵恢,它會每15分鐘喚醒app 。
? 這種方式的定位你就需要調(diào)用startMonitoringSignificantLocationChanges 而不是startUpdatingLocation且一定是NSLocationAlwaysUsageDescription的定位描述? 官方代碼
- (void)startSignificantChangeUpdates
{
// Create the location manager if this object does not
// already have one.
if (nil == locationManager)
locationManager = [[CLLocationManager alloc] init];
locationManager.delegate = self;
[locationManager startMonitoringSignificantLocationChanges];
}
如果想要停止 調(diào)用stopMonitoringSignificantLocationChanges? 定位接受到的定位信息都在這個方法里面locationManager:didUpdateLocations:? 定位失敗了調(diào)用locationManager:didFailWithError:
注意:這種方式的定位是不需要設置后臺模式圾结,就可以后臺拿到定位數(shù)據(jù)的 處理事務只有10s時間(如果想需要更多時間來處理你的業(yè)務用beginBackgroundTaskWithName:expirationHandler:(這個方法會讓你有10分鐘的時間來處理你的業(yè)務)) 即使你的app切換到后臺了掛起了或者后臺了瑰剃,這種方式定位如果接受到定位消息會自動重新啟動或喚醒app(官方說If your app is terminated either by a user or by the system, the system doesn’t automatically restart your app when new location updates arrive. A user must explicitly relaunch your app before the delivery of location updates resumes.The only way to have your app relaunched automatically is to use region monitoring or the significant-change location service.)齿诉,只不過需要你在后臺重新創(chuàng)建CLLocationManager手動重新開啟定位startMonitoringSignificantLocationChanges? 不過如果用戶設置app后臺刷新Background App Refresh 不開啟的話 有定位消息來的通知的時候筝野,系統(tǒng)就不會重新喚醒或重啟app晌姚。Background App Refresh 不開啟 即使在前臺,這種方式的定位將接受不到定位消息歇竟。也就是 這種方式的定位要保證Background App Refresh 是開啟的狀態(tài) ?獲取后臺刷新狀態(tài)方法是backgroundRefreshStatusproperty of theUIApplicationclass.你可以在定位之前先判斷下挥唠,然后給出提示。
其實后臺定位模式焕议,有很多優(yōu)化的方式來節(jié)省電池的電量宝磨,可以查看官方文檔。