成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

后臺定位上傳的代碼實踐

移動開發
現在做的是LBS定位的社交APP 其中主要的一個功能就是能夠實時定位社交圈中各個成員的位置 后臺實時上傳位置則是非常重要的一個技術點 接下來就來說說我關于這方面的實踐經驗

[[141852]]

前言

之前的文章說過 我現在做的是LBS定位的社交APP 其中主要的一個功能就是能夠實時定位社交圈中各個成員的位置 后臺實時上傳位置則是非常重要的一個技術點 接下來就來說說我關于這方面的實踐經驗

需求

先來看看實現這個功能的具體需求是什么 由于我們是實時定位的生活類社交APP 所以我們需要做到一下幾點

1. 如果用戶的位置在持續變化 則隔一段時間上報一次

  由于我們希望能夠實時的將用戶的位置變化反饋在APP里 所以定時的上報是剛需

2. 如果用戶的移動速度很慢 則隔一段距離上報一次

  如果用戶是低速率的狀態(比如步行的移動速度大概就是1m/s左右) 這個時候如果還按(1)中的方式來上報的話 由于變化太小 地圖上的點會非常的密集 這種數據的意義不大(而且如果要做軌跡服務的話 這些密集點都是必須有花掉的) 所以這時候我們按照距離間隔來上報

3. 如果用戶的位置在到達某處后沒有變化 則不繼續上報

  我們只關心位置的變化 如果用戶的位置沒有變化或者變化很小 其實是不需要上報其位置的(比如進入的公司 或者等一個很長時間的紅燈) 這時候我們就不上報(以達到省電的目的)

4. 切換到后臺也要能定位上報

  后臺上報是必須的 用戶不可能一直運行著我們的APP (iOS4開始就支持了)

5. APP因各種原因終止運行后(用戶主動關閉, 系統殺掉) 也要能定位上報

用戶主動關閉APP的幾率不大 但是因系統調度被殺掉的情況是很普遍的 這個時候我們也要能夠上報 (iOS7開始已支持被殺掉后喚醒)

分析完需求 接下來就開始介紹如何實現

準備

首先做一些準備工作

在target的Capabilities選項中打開Background Modes 并勾選Location updates

然后在plist中添加NSLocationAlawaysUsageDescription的鍵 在value中隨便鍵入任何內容


完成這兩步 我們的前期工作就完成了 Background Modes是iOS7帶入的新功能 而NSLocationAlawaysUsageDescription為了增強權限機制引入的提示描述 不添加這個的話 定位功能可是使用不了的

代碼

定位肯定要跟CLLocationManager打交道 所以我們先定義一個CLLocationManager的子類 并根據需求中的幾點定義三個變量

  1. @interface MMLocationManager : CLLocationManager 
  2.  
  3. + (instancetype)sharedManager; 
  4.  
  5. @property (nonatomic, assign) CGFloat minSpeed;     //最小速度 
  6. @property (nonatomic, assign) CGFloat minFilter;    //最小范圍 
  7. @property (nonatomic, assign) CGFloat minInteval;   //更新間隔 
  8.  
  9. @end 

 這里解釋一下這幾個參數

  • minSpeed 如果當前運動速度大于此值 則滿足需求(1) 以時間為更新依據(minFilter) 如果當前運動速度小于此值 則滿足需求(2) 以范圍為更新依據(minInteval)
  • minFilter 最小的觸發范圍 用于需求(1)
  • minInteval 更新間隔 用于需求(2)

接下來是初始化函數

  1. - (instancetype)init 
  2.     self = [super init]; 
  3.     if ( self ) 
  4.     { 
  5.         self.minSpeed = 3
  6.         self.minFilter = 50
  7.         self.minInteval = 10
  8.  
  9.         self.delegate = self; 
  10.         self.distanceFilter  = self.minFilter; 
  11.         self.desiredAccuracy = kCLLocationAccuracyBest; 
  12.     } 
  13.     return self; 

這里的默認值可以根據需求來調整

然后是位置更新后的處理邏輯 其實也非常的簡單、

  1. - (void)locationManager:(CLLocationManager *)manager didUpdateLocations:(NSArray *)locations 
  2.     CLLocation *location = locations[0]; 
  3.  
  4.     NSLog(@"%@",location); 
  5.  
  6.     //根據實際情況來調整觸發范圍 
  7.     [self adjustDistanceFilter:location]; 
  8.  
  9.     //上傳數據 
  10.     [self uploadLocation:location]; 

而這個adjustDistanceFilter函數 就是整個代碼的核心 會根據當前速度來動態的調整distanceFilter這個參數 以滿足我們的需求 

  1. /** 
  2. *  規則: 如果速度小于minSpeed m/s 則把觸發范圍設定為50m 
  3. *  否則將觸發范圍設定為minSpeed*minInteval 
  4. *  此時若速度變化超過10% 則更新當前的觸發范圍(這里限制是因為不能不停的設置distanceFilter, 
  5. *  否則uploadLocation會不停被觸發) 
  6. */ 
  7. - (void)adjustDistanceFilter:(CLLocation*)location 
  8. //    NSLog(@"adjust:%f",location.speed); 
  9.      
  10.     if ( location.speed < self.minSpeed ) 
  11.     { 
  12.         if ( fabs(self.distanceFilter-self.minFilter) > 0.1f ) 
  13.         { 
  14.             self.distanceFilter = self.minFilter; 
  15.         } 
  16.     } 
  17.     else 
  18.     { 
  19.         CGFloat lastSpeed = self.distanceFilter/self.minInteval; 
  20.          
  21.         if ( (fabs(lastSpeed-location.speed)/lastSpeed > 0.1f) || (lastSpeed < 0) ) 
  22.         { 
  23.             CGFloat newSpeed  = (int)(location.speed+0.5f); 
  24.             CGFloat newFilter = newSpeed*self.minInteval; 
  25.              
  26.             self.distanceFilter = newFilter; 
  27.         } 
  28.     } 

這里要注意到的是distanceFilter這個參數不能一直進行設置 因為每次設置完以后 再接下來的一秒以后 會立即觸發didUpdateLocations回調(系統的標準最短更新間隔是1秒 即更新頻率為1hz) 所以這里只有當變化超過10%的時候才會重置distanceFilter

#p#

接下來 為了能夠正確的在被殺掉的情況下被喚醒 我們還要做***一步操作 在AppDelegate的didFinishLaunchingWithOptions中加入下面的代碼

  1. if ([launchOptions objectForKey:UIApplicationLaunchOptionsLocationKey]) { 
  2.      
  3.     if ( [[MMLocationManager sharedManager] respondsToSelector:@selector(requestAlwaysAuthorization)] ) 
  4.     { 
  5.         [[MMLocationManager sharedManager] requestAlwaysAuthorization]; 
  6.     } 
  7.  
  8.     //這是iOS9中針對后臺定位推出的新屬性 不設置的話 可是會出現頂部藍條的哦(類似熱點連接) 
  9.     if ( [self respondsToSelector:@selector(allowsBackgroundLocationUpdates)]) 
  10.     { 
  11.         [MMLocationManager sharedManager].allowsBackgroundLocationUpdates = YES; 
  12.     } 
  13.  
  14.     [[MMLocationManager sharedManager] startUpdatingLocation]; 

這是因為被殺掉的APP 在后臺被系統喚醒時 launchOptions會包含UIApplicationLaunchOptionsLocationKey**字段來進行標識 這時我們再重新啟動定位功能即可

至此 滿足我們需求的定位功能就完成了 為此我寫了一個demo來驗證(使用模擬器 然后選擇Debug->Location->Freeway Drive) 結果如下

接下來我們會討論一下相關的幾個問題

討論

為什么不用定時器來控制定位間隔

網上有很多教程是用NSTimer來實現的 但是其實這樣不是很好 雖然定位的間隔是固定的 但是耗電的問題無法解決 后臺會持續的更新定位 無論當前的位置是否在更新 當然 如果你的使用場景就是要每隔一段時間來上傳 就可以使用定時器來處理

使用distanceFilter來處理 會有些什么問題

由于distanceFilter=currentSpeed*minInteval 那么間隔的時間因為速度的變化而會有波動 但是這個波動是在可接受范圍的 如果速度加快或者變慢 那么下一次的更新時間則會相應的縮短或者變長 但是因為我們是在真實生活環境中 速度的變化不可能那么快 所以這個誤差是可以接受的 另外我們對distanceFilter針對速度進行矯正 因而總體來說 間隔還是會保持在我們與其的范圍內的

為什么不使用allowDeferredLocationUpdatesUntilTraveled:timeout:

allowDeferredLocationUpdatesUntilTraveled是iOS6推出的一個新的API 看名字我們可以知道這個函數的作用是延遲位置更新 直到移動了xx米或者時間超過了xx秒 那么這個函數不正好滿足了我們的所有要求么? 可是萬萬沒想到 事情并不是這樣的 這個函數并不好用

接下來是吐槽時間 ლ(⁰⊖⁰ლ)

為什么說這個函數不好用呢? 首先 這個函數的要求很多 我們來看看要這個函數起作用要滿足哪些條件

  • 必須iPhone5以及之后的硬件設備才支持
  • desiredAccuracy必須設置為kCLLocationAccuracyBest或者kCLLocationAccuracyBestForNavigation
  • distanceFilter必須設置為kCLDistanceFilterNone
  • 只在APP運行在后臺時生效 前臺運行時是不會進行延遲處理的
  • 只有系統在低功耗(Low Power State)的時候才有可能生效

關于Low Power State在iOS中的描述 我只在蘋果官網的文檔中找到部分定義

iOS is very good at getting a device into a low power state when it’s not being used. At idle, very little power is drawn and energy impact is low. When tasks are actively occurring, system resources are being used and those resources require energy. However, sporadic tasks can cause the device to enter an intermediate state—neither idle nor active—when the device isn’t doing anything. There may not be enough time during these intermediate states for the device to reach absolute idle before the next task executes. When this occurs, energy is wasted and the user’s battery drains faster.

據我簡單的了解 這個**Low Power State”只有在黑屏的狀態下(不只是鎖屏)才有可能觸發 只要有任何電量屏幕的操作(就連推送也算) 都會使APP退出這個狀態 同時 如果在充電狀態下 也是無法進入的

我嘗試在真機和模擬器上使用這個API 但結果APP還是以1HZ的頻率在定位(設置了kCLDistanceFilterNone的原因)
雖然locationManager:didFinishDeferredUpdatesWithError:在指定的時間后成功的回調了 但是結果還是沒有deffer 于是我查了一下 原來這個函數無法直接進行調試的 因為:

  • 不支持模擬器 deferredLocationUpdatesAvailable用于檢測設備是否支持 模擬器會返回NO
  • 不支持真機調試 因為調試時Xcode會阻止程序休眠 導致程序無法進入低功耗狀態

結論就是…這個東西連調試都沒辦法 所以我也沒有那么多時間跑到外面去測試這個東西… 況且使用我上述的方法已經基本可以滿足需求… 所以我已放棄繼續研究這個API了 因為就算使用了這個東西 也僅僅是錦上添花而已

如果有哪些同學知道如何正確的使用這個東西 請留言告訴我 萬分感謝!

責任編輯:倪明 來源: 里脊串的開發隨筆
相關推薦

2015-07-27 09:39:24

后臺代碼

2023-11-06 09:56:10

研究代碼

2012-09-29 10:09:19

網站架構后臺構建架構

2018-01-16 16:32:12

架構前臺后臺

2023-01-05 07:54:49

vivo故障定位

2022-05-24 16:14:01

CSS實踐

2015-06-24 10:06:09

iOS 9適配后臺

2014-10-29 13:52:38

程序員

2023-04-28 08:06:04

低代碼AI智能

2020-03-09 14:10:48

代碼開發工具

2021-09-08 09:37:54

工具代碼Python

2020-05-25 11:14:59

代碼程序開發

2017-05-12 09:24:21

Python代碼Logger

2012-08-09 09:10:56

代碼審查代碼

2015-09-23 10:14:48

iOS 代碼實踐

2015-01-29 09:52:43

Swift 開源蘋果

2015-08-13 09:39:41

2018-05-10 15:06:43

Java Web分層實踐

2018-01-12 14:37:34

Java代碼實踐

2014-05-16 10:51:33

科學代碼最佳實踐
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 国产午夜精品一区二区三区嫩草 | 操皮视频 | 亚洲视频一区在线观看 | 日韩欧美视频免费在线观看 | 国产激情偷乱视频一区二区三区 | 久久久久国产精品 | 国产午夜精品久久 | 日韩一级免费电影 | 亚洲国产视频一区 | 男女又爽又黄视频 | 欧美精品片| 在线观看国产视频 | 99re6在线视频精品免费 | 国产成人免费 | 视频一区二区在线观看 | 亚洲欧美在线视频 | 伊人久久精品一区二区三区 | 国产欧美日韩综合精品一区二区 | 天堂一区在线观看 | 欧美视频网| 三级视频在线观看电影 | 粉嫩一区二区三区国产精品 | 美女视频一区 | 日韩久久久一区二区 | 91国产精品| 久久精品一区二区三区四区 | 成在线人视频免费视频 | 亚洲欧美日韩高清 | 91pron在线| 亚洲国产成人精品久久 | 中文字幕在线免费 | 2019精品手机国产品在线 | 欧美日韩综合 | 国产偷久久一级精品60部 | 九九色综合| 国产一二三视频在线观看 | 精品国产亚洲一区二区三区大结局 | 在线成人一区 | 91久久| 九九九精品视频 | 国产精品久久 |