來源:Getting the User’s Location
- 標(biāo)準(zhǔn)位置服務(wù),提供了高度可配置的方式來獲取當(dāng)前位置并且跟蹤改變.
- 區(qū)域監(jiān)控讓你可以監(jiān)控定義的地理區(qū)域和藍(lán)牙信標(biāo).(藍(lán)牙只在iOS上有效)
- The significant-change location service提供了當(dāng)大的位置變動時,通知并得到當(dāng)前位置.正確的使用它可以避免免費太多的電量.
要求在iOS App里使用定位服務(wù)
在app里的info.plist
文件里包含UIRequiredDeviceCapabilities
鍵.App Store用在這個鍵里的信息來確定是否可以被下載,如果沒有列表里的功能就不能被下載.
UIRequiredDeviceCapabilities
的值,是標(biāo)識功能的字符串所組成的數(shù)組,這些功能應(yīng)該是被app所需要的.有兩個相關(guān)定位服務(wù)的字符串:
- 如果你需要定位服務(wù),請包含
location-services
字符串. - 如果您的應(yīng)用需要GPS硬件提供的精度晓淀,請包含gps字符串.
重要:如果你的iOS app在沒有上述的字符串也能正確使用定位服務(wù),那就不在
UIRequiredDeviceCapabilities
包含它們
獲得用戶的當(dāng)前位置
Core Location framework 可以讓你位置設(shè)備的當(dāng)前位置并在app里使用它們.你可以配置定位服務(wù),讓framework來報告設(shè)備的位置,也可以周期性的更新數(shù)據(jù).
兩個服務(wù)可以讓你得到用戶的當(dāng)前位置:
- 標(biāo)準(zhǔn)定位服務(wù):得到定位數(shù)據(jù),并以指定的精度水平跟蹤定位的改變
- significant-change定位服務(wù):只有當(dāng)在設(shè)備有重大的位置改變時才會通知,比如說500或以上.
收集定位信息是一項很耗電量的操作.你應(yīng)該選擇合適的定位服務(wù),避免消耗電量.舉例:
- 如果你的iOS app即使在后臺也要持續(xù)的跟蹤定位,使用準(zhǔn)備定位服務(wù)并指明
UIBackgroundModes
,讓app可以在后臺也可繼續(xù)的運行和接收定位數(shù)據(jù).(在這種情況下,你應(yīng)該保證設(shè)置 location manager的pausesLocationUpdatesAutomatically
屬性為true
,來節(jié)省電量.) - 如果你不需要GPS級別的精度,并且你不需要持續(xù)的跟蹤,你可以使用significant-change定位服務(wù).正確的使用這個服務(wù)是非常重要的,因為你至少15分鐘會喚醒系統(tǒng)和你的app,即使沒有位置的改變,并且在你停止它之前會持續(xù)的運行.
查看定位服務(wù)是否有效
以下幾點會讓定位無效:
- 用戶在Settings或系統(tǒng)偏好里關(guān)閉了定位服務(wù).
- 用戶拒絕和指定app的定位服務(wù).
- 設(shè)備在飛行模式下,無法啟動必要的設(shè)備
基于以上的理由,在你要開始定位服務(wù)之前,你都要查看一下CLLocationManager
的類方法locationServicesEnabled
確保定位服務(wù)是開啟的.如果返回的是false
,你又要開始定位服務(wù),系統(tǒng)會讓用戶確認(rèn)定位服務(wù)是否應(yīng)該被開啟.
開始標(biāo)準(zhǔn)定位服務(wù)
準(zhǔn)備定位服務(wù)是得到用戶位置的很普遍的方式,因為它在所有的iOS和OS X設(shè)備都有效.在你使用這個服務(wù)之前,你要配置定位數(shù)據(jù)所需的精度在移動要新位置之前所需的距離.當(dāng)你開始服務(wù)之前,它會使用特定的屬性去查明硬件是否有效,然后會持續(xù)的報告定位事件給你的app.這項服務(wù)適合于導(dǎo)航app,或高精度定位或周期性的更新流.
為了使用準(zhǔn)備定位服務(wù),創(chuàng)建CLLocationManager
類的一個實例,并且配置的desireAccuracy
和distanceFilter
屬性.賦值了個delegate
并調(diào)用startUpdatingLoation
方法去開始定位通知.當(dāng)定位數(shù)據(jù)變成有效時,location manager就會通知它的delegate
對象.當(dāng)定位更新已經(jīng)開始遞送時,你可直接在CLLocationManager
對象獲取最新的數(shù)據(jù),而不用等新事件的遞送.停止這種定位的更新,調(diào)用location manager對象的stopUpdatingLocation
方法.
- (void)startStandardUpdates
{
// Create the location manager if this object does not
// already have one.
if (nil == locationManager)
locationManager = [[CLLocationManager alloc] init];
locationManager.delegate = self;
locationManager.desiredAccuracy = kCLLocationAccuracyKilometer;
// Set a movement threshold for new events.
locationManager.distanceFilter = 500; // meters
[locationManager startUpdatingLocation];
}
開始 Significant-Change 定位服務(wù)
這個服務(wù)需要
kCLAuthorizationStatusAuthorizedAlways
授權(quán)狀態(tài).
- (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];
}
停止服務(wù)
[locationManager stopMonitoringSignificantLocationChanges];
如果你的iOS app被掛起或終止,當(dāng)新的數(shù)據(jù)來的時候,這個服務(wù)會自動喚醒你的app.在喚醒時候內(nèi),app會進(jìn)入后臺并且給你一小段時間(大概10秒)去手動重啟定位服務(wù)并處理定位數(shù)據(jù).(在任何定位遞送之前你必須要重啟定位服務(wù)Knowing When to Start Location Services),因為你的app是在后臺的,你必須做最少的事情,并且避免做任何在預(yù)計時間內(nèi)得不到答案的任務(wù)(比如:network).否則,你的app會被終止.如里你需要更多的后臺運行時間去處理位置數(shù)據(jù),它可以用UIApplication
類的beginBackgroundTaskWithName:expirationHandler:
去請求很多的后臺運行時間.
當(dāng)用戶不管是全局還是在你的app里關(guān)閉了后臺app刷新,significant-change定位服務(wù)不會重啟你的app.更進(jìn)一步,即使你在前臺,app也不會接收到significant-change或區(qū)域跟蹤事件.
從服務(wù)器那里接收定位數(shù)據(jù)
不論是準(zhǔn)備定位服務(wù)還是significant-change定位服務(wù),接收定位事件方式是一樣的.location manager是將事件報告給delegate
的locationManager:didUpdateLocations:
方法.如果有接收事件是有錯誤,location manager會調(diào)用delegate
的locationManager:didFailWithError:
方法.
因為location manager有時會返回緩存的事件(就是不是最新的),最好是檢查一下定位的時間戳:
// Delegate method from the CLLocationManagerDelegate protocol.
- (void)locationManager:(CLLocationManager *)manager
didUpdateLocations:(NSArray *)locations {
// If it's a relatively recent event, turn off updates to save power.
CLLocation* location = [locations lastObject];
NSDate* eventDate = location.timestamp;
NSTimeInterval howRecent = [eventDate timeIntervalSinceNow];
//這是15秒之外的就是舊數(shù)據(jù),如果做導(dǎo)航的話可以更短.
if (abs(howRecent) < 15.0) {
// If the event is recent, do something with it.
NSLog(@"latitude %+.6f, longitude %+.6f\n",
location.coordinate.latitude,
location.coordinate.longitude);
}
}
使用時間戳名,還可以用精度來判斷.
知道什么時候要開始定位服務(wù)
使用定位服務(wù)的app,不會自動開始這些服務(wù).有一些例外情況,不要一些情況下開始定位服務(wù),比如在啟動的時候或者因一些原因定位服務(wù)不能被使用的時候.
一個app要使用Core Location必須要在info.plist
里包含NSLocationAlwaysUsageDescription
或NSLocationWhenInUseUsageDescription
鍵,并且設(shè)置一些字符串給它們,說明怎么使用定位數(shù)據(jù).如果在你調(diào)用requestWhenInUseAuthorization
方法,但你沒有設(shè)置以上那些鍵,系統(tǒng)就會忽略你的這些請求.
如果你在監(jiān)控區(qū)域或使用significant-change定位服務(wù),有一些情況是你必須在啟動時候要開啟定位服務(wù)的.可以終止使用這些服務(wù)的app喻喳,并在新的位置事件到達(dá)時隨后重新啟動.app自己重新啟動的時候,定位服務(wù)不會自動開啟.當(dāng)你一個app因為定位更新而重新啟動時,啟動選項dictionary會被傳到application:willFinishLaunchingWithOptions:
或application:didFinishLaunchingWithOptions:
里,使用的是UIApplicationLaunchOptionsLocationKey
鍵.該鍵的存在表明新的位置數(shù)據(jù)正在等待傳送到您的應(yīng)用程序.為了獲取這些數(shù)據(jù),你必須創(chuàng)建新的CLLocationManager
對象并重新開啟定位服務(wù).當(dāng)你開始這些定位服務(wù)時,location manager會把所以待更新的定位數(shù)據(jù)發(fā)送給delegate
.
在后臺獲取定位事件(iOS only)
... ...
當(dāng)你的app在后臺時,推遲定位更新
... ...