轉(zhuǎn)載(http://www.cnblogs.com/oshushu/p/4569252.html)
1.概述
由于公司一款產(chǎn)品的需求,最近一直在研究iOS設(shè)備的后臺(tái)定位。主要的難點(diǎn)就是,當(dāng)系統(tǒng)進(jìn)入后臺(tái)之后,程序會(huì)被掛起,屆時(shí)定時(shí)器、以及代碼都不會(huì)Run~ 所以一旦用戶將我的App先換到了后臺(tái),我的定位功能將無法繼續(xù)。
經(jīng)過了我?guī)滋斓牟檎屹Y料和嘗試,我發(fā)現(xiàn)了一個(gè)我個(gè)人認(rèn)為非常簡(jiǎn)單的方法來解決這個(gè)問題。這個(gè)方法說白了是使用一個(gè)第三方的類庫(kù),經(jīng)過測(cè)試,App在真機(jī)后臺(tái)運(yùn)行3小時(shí),App依然在定時(shí)的向服務(wù)器發(fā)送位置坐標(biāo)。這個(gè)類庫(kù)的名字叫做“voyage11/Location”,作者的叫做Ricky。大家可以去Github下載這個(gè)類庫(kù)。要注意的時(shí),要測(cè)試后臺(tái)定位功能,最好在真機(jī)上測(cè)試,模擬器上測(cè)試怪怪的~結(jié)果不要作為參考。
下載后我們得到的是一個(gè)工程,大家運(yùn)行一下看看效果,UI什么也沒有,效果都顯示在控制臺(tái)里面,運(yùn)行一會(huì)之后,切換的后臺(tái)再看看效果。
2.怎么用-voyage類庫(kù)的基本類/方法
下面介紹一下這個(gè)類庫(kù)的類和方法,了解了這些之后,你大概就會(huì)知道怎么使用voyage/Location這個(gè)類庫(kù)了。
從下載的工程里,我可以直觀的看到這個(gè)類庫(kù)的結(jié)構(gòu):
千萬不要感覺這么一大坨會(huì)不會(huì)很麻煩TT NO!不要害怕,他用起來真的很簡(jiǎn)單,你只需要略微修改幾個(gè)參數(shù),其他的那一坨你可以不用管(如果只求能用,不求甚解的話)。
我來簡(jiǎn)單說一下這幾個(gè)類的作用:
LocationTracker & Other
和我們直接打交道的主要就是LocationTracker這個(gè)類。用這個(gè)類,我們可以配置定位的相關(guān)參數(shù)。我們來看看這個(gè)類的主要方法:
構(gòu)造方法,獲得一個(gè)LocationTraker的單例對(duì)象(不了解單列是啥意思的,你可以理解成創(chuàng)建一個(gè)全局變量)。
+ (CLLocationManager *)sharedLocationManager;
這個(gè)方法是開始追蹤定位,之后,定位功能就跑起來了。
1 - (void)startLocationTracking;
這個(gè)方法和上面的方法是一對(duì),它用來關(guān)閉定位追蹤。
- (void)stopLocationTracking;
這個(gè)方法用來向服務(wù)器發(fā)送已獲取的設(shè)備位置信息。
- (void)updateLocationToServer;
另外還有兩個(gè)類是“LocationShareModel”和“BackgroundTaskManager”。他們的工作主要是處理定位服務(wù)的后臺(tái)運(yùn)行和處理設(shè)備獲取的定位數(shù)據(jù)。具體的原理我們不用去管它。
That's all怎么樣,真的很簡(jiǎn)單吧
-
示例
好啦,趁熱乎,我們趕緊拿來用用試試吧~
首先我們把我們要用到的類先從下載的項(xiàng)目文件夾中拿出來,我們要用的總共有三個(gè)類 :“LocationTracker”“LocationShareModel”和“BackgroundTaskManager”如下圖:
下一步,Xcode打開我們要使用這個(gè)類庫(kù)的工程,把這三個(gè)類庫(kù)加入到工程中去(你可以選中這6個(gè)文件拖進(jìn)文件導(dǎo)航)
拋開這個(gè)類庫(kù)不談,如果要進(jìn)行后臺(tái)定位服務(wù),你需要確保為工程做出如下設(shè)置:
1.開啟后臺(tái)定位模式:選中工程Target->Capabilities->Background Modes-勾選Location updates:
2.在Plist中添加前/后臺(tái)定位的鍵值:在Plist根目錄新建兩個(gè)鍵值如下,這些鍵值將會(huì)在程序開啟時(shí)讓用戶允許開啟后前/臺(tái)定位。
設(shè)置完以上配置之后,我們就可以來想用我們的voyageLocation啦
首先在你想要使用定位功能的ViewController 導(dǎo)入頭文件
#import "LocationTracker.h"
然后聲明兩個(gè)成員變量:
@property LocationTracker * locationTracker;
@property (nonatomic) NSTimer* locationUpdateTimer;
之后寫一個(gè)方法配置LocationTraker:
-(void)setUpLocationTraker{
self.locationTracker = [[LocationTracker alloc]init];
[self.locationTracker startLocationTracking];
//設(shè)定向服務(wù)器發(fā)送位置信息的時(shí)間間隔
NSTimeInterval time = 300.0;
//開啟計(jì)時(shí)器
self.locationUpdateTimer =
[NSTimer scheduledTimerWithTimeInterval:time
target:self
selector:@selector(updateLocation)
userInfo:nil
repeats:YES];
}
上面計(jì)時(shí)器每隔300s運(yùn)行一次“updateLocation”方法,該方法的實(shí)現(xiàn)如下:
-(void)updateLocation {
NSLog(@"開始獲取定位信息...");
//向服務(wù)器發(fā)送位置信息
[self.locationTracker updateLocationToServer];
}
上面的updateLocationToServer方法就是你向服務(wù)器發(fā)送信息的方法了,這個(gè)方法需要你依照自己的需求進(jìn)行改動(dòng)打開“LocationTraker.m”文件找到該方法:
- (void)updateLocationToServer {
NSLog(@"updateLocationToServer");
// Find the best location from the array based on accuracy
NSMutableDictionary * myBestLocation = [[NSMutableDictionary alloc]init];
for(int i=0;i<self.shareModel.myLocationArray.count;i++){
NSMutableDictionary * currentLocation = [self.shareModel.myLocationArray objectAtIndex:i];
if(i==0)
myBestLocation = currentLocation;
else{
if([[currentLocation objectForKey:ACCURACY]floatValue]<=[[myBestLocation objectForKey:ACCURACY]floatValue]){
myBestLocation = currentLocation;
}
}
}
NSLog(@"My Best location:%@",myBestLocation);
//If the array is 0, get the last location
//Sometimes due to network issue or unknown reason, you could not get the location during that period, the best you can do is sending the last known location to the server
if(self.shareModel.myLocationArray.count==0)
{
NSLog(@"Unable to get location, use the last known location");
self.myLocation=self.myLastLocation;
self.myLocationAccuracy=self.myLastLocationAccuracy;
}else{
CLLocationCoordinate2D theBestLocation;
theBestLocation.latitude =[[myBestLocation objectForKey:LATITUDE]floatValue];
theBestLocation.longitude =[[myBestLocation objectForKey:LONGITUDE]floatValue];
self.myLocation=theBestLocation;
self.myLocationAccuracy =[[myBestLocation objectForKey:ACCURACY]floatValue];
}
NSLog(@"Send to Server: Latitude(%f) Longitude(%f) Accuracy(%f)",self.myLocation.latitude, self.myLocation.longitude,self.myLocationAccuracy);
//TODO: 在這里插入你向服務(wù)器發(fā)送請(qǐng)求的代碼
//當(dāng)你向服務(wù)器發(fā)送位置信息成功后,要清空當(dāng)前的數(shù)組,以便下一回合的定位
[self.shareModel.myLocationArray removeAllObjects];
self.shareModel.myLocationArray = nil;
self.shareModel.myLocationArray = [[NSMutableArray alloc]init];
}
在上面代碼的第處40行進(jìn)行修改,添加你像服務(wù)器發(fā)送位置信息的請(qǐng)求,當(dāng)請(qǐng)求成功后,不要忘記執(zhí)行第43-45行的代碼,清空數(shù)組,以便下一次定位。
例如我加入的代碼如下,我是用了AFNetworking的網(wǎng)絡(luò)請(qǐng)求類庫(kù):
AFHTTPRequestOperationManager *manager=[AFHTTPRequestOperationManager manager];
NSString *url=[NSString stringWithFormat:@"http://172.1.1.36:8080/uploadDeviceLocation.action"];
NSMutableDictionary *parameter=[[NSMutableDictionary alloc]init];
[parameter setObject:@"####################" forKey:@"udid"];
[parameter setObject: [NSString stringWithFormat:@"%f",self.myLocation.longitude] forKey:@"x"];
[parameter setObject:[NSString stringWithFormat:@"%f",self.myLocation.latitude] forKey:@"y"];
[manager GET:url parameters:parameter success:^(AFHTTPRequestOperation *operation, id responseObject) {
NSLog(@" 成功了");
} failure:^(AFHTTPRequestOperation *operation, NSError *error) {
NSLog(@"失敗了");
[self.shareModel.myLocationArray removeAllObjects];
self.shareModel.myLocationArray = nil;
self.shareModel.myLocationArray = [[NSMutableArray alloc]init];
}];
OK~ 搞定,趕緊試試吧! 哦對(duì)了,你不覺得你忘記什么了嗎? 對(duì)了 要把 [self setUpLocationTraker] 方法放到你的 viewDidLoad 里面~哈哈
這樣 后臺(tái)位置上傳就解決了。這是控制臺(tái)打出的Log。
4.總結(jié)
解決了糾結(jié)好幾天的問題,現(xiàn)在我的心里還有點(diǎn)小興奮。總結(jié)一下這個(gè)類庫(kù)的特點(diǎn),第一就是使用非常簡(jiǎn)單。第二,運(yùn)行穩(wěn)定,經(jīng)過我近2個(gè)小時(shí)的測(cè)試,定位一直跑,后臺(tái)一直能收到上報(bào)的信息,妥妥的。第三,這個(gè)類庫(kù)的作者考慮到了定位耗電的問題,我在測(cè)試時(shí),用的是一部很老的iPhone4S,兩個(gè)小時(shí)掉了10%的電,對(duì)于我來說還是可以接受的。再次感謝Rickey。這是他的博客,下面有捐款的鏈接,希望大家去表示一下對(duì)他的感謝(支持paypal、visa、master等,銀聯(lián)不支持哦)。