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