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

后臺定位上傳的代碼實踐

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

[[142068]]

前言

之前的文章說過 我現在做的是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. + (instancetype)sharedManager; 
  3. @property (nonatomic, assign) CGFloat minSpeed; //最小速度 
  4. @property (nonatomic, assign) CGFloat minFilter; //最小范圍 
  5. @property (nonatomic, assign) CGFloat minInteval; //更新間隔 
  6. @end 

這里解釋一下這幾個參數

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

minFilter 最小的觸發范圍 用于需求(1)

minInteval 更新間隔 用于需求(2)

接下來是初始化函數

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

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

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

  1. - (void)locationManager:(CLLocationManager *)manager didUpdateLocations:(NSArray *)locations 
  2. CLLocation *location = locations[0]; 
  3. NSLog(@"%@",location); 
  4. //根據實際情況來調整觸發范圍 
  5. [self adjustDistanceFilter:location]; 
  6. //上傳數據 
  7. [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. if ( fabs(self.distanceFilter-self.minFilter) > 0.1f ) 
  12. self.distanceFilter = self.minFilter; 
  13. else 
  14. CGFloat lastSpeed = self.distanceFilter/self.minInteval; 
  15.  
  16. if ( (fabs(lastSpeed-location.speed)/lastSpeed > 0.1f) || (lastSpeed < 0) ) 
  17. CGFloat newSpeed = (int)(location.speed+0.5f); 
  18. CGFloat newFilter = newSpeed*self.minInteval; 
  19.  
  20. self.distanceFilter = newFilter; 

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

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

  1. if ([launchOptions objectForKey:UIApplicationLaunchOptionsLocationKey]) { 
  2.  
  3. if ( [[MMLocationManager sharedManager] respondsToSelector:@selector(requestAlwaysAuthorization)] ) 
  4. [[MMLocationManager sharedManager] requestAlwaysAuthorization]; 
  5. //這是iOS9中針對后臺定位推出的新屬性 不設置的話 可是會出現頂部藍條的哦(類似熱點連接) 
  6. if ( [self respondsToSelector:@selector(allowsBackgroundLocationUpdates)]) 
  7. [MMLocationManager sharedManager].allowsBackgroundLocationUpdates = YES; 
  8. [[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了 因為就算使用了這個東西 也僅僅是錦上添花而已

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

小結

文中的demo可以在這里找到 另外demo中用到了Realm來存儲數據(模擬上傳操作) 有興趣的同學可以看一下

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

2015-07-23 10:09:45

后臺定位代碼

2023-11-06 09:56:10

研究代碼

2012-09-29 10:09:19

網站架構后臺構建架構

2023-01-05 07:54:49

vivo故障定位

2018-01-16 16:32:12

架構前臺后臺

2022-05-24 16:14:01

CSS實踐

2015-06-24 10:06:09

iOS 9適配后臺

2023-04-28 08:06:04

低代碼AI智能

2020-03-09 14:10:48

代碼開發工具

2014-10-29 13:52:38

程序員

2021-09-08 09:37:54

工具代碼Python

2012-08-09 09:10:56

代碼審查代碼

2015-09-23 10:14:48

iOS 代碼實踐

2020-05-25 11:14:59

代碼程序開發

2017-05-12 09:24:21

Python代碼Logger

2015-01-29 09:52:43

Swift 開源蘋果

2018-01-12 14:37:34

Java代碼實踐

2015-08-13 09:39:41

2018-05-10 15:06:43

Java Web分層實踐

2014-05-16 10:51:33

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

51CTO技術棧公眾號

主站蜘蛛池模板: 成人在线中文字幕 | 国内精品一区二区三区 | 日韩欧美一级精品久久 | 狠狠的操 | m豆传媒在线链接观看 | 国产不卡在线 | 色网在线看 | 国产精品成人一区二区三区吃奶 | 亚洲视频一区二区三区 | 一区二区三区四区五区在线视频 | 精品粉嫩超白一线天av | 亚洲精品一区二区三区在线 | 日韩精品免费在线观看 | 亚洲高清视频一区二区 | 精品国产一区二区三区久久久蜜月 | 99精品一区二区 | 精品在线一区二区 | 国色天香成人网 | 中文字幕成人av | 91看片视频| 一区二区三区四区国产精品 | 欧美一级在线观看 | 亚洲精品女人久久久 | 国产不卡一区在线观看 | 亚洲福利视频一区二区 | 九九伦理片 | 香蕉久久a毛片 | 99精品久久99久久久久 | 激情欧美一区二区三区中文字幕 | 中文在线视频观看 | 国产亚洲一区二区精品 | 国产一二三区免费视频 | 国产高清一区二区 | 日韩在线免费看 | 51ⅴ精品国产91久久久久久 | 精品一区二区观看 | 精品国产一区二区三区久久狼黑人 | 91精品国产综合久久久久久蜜臀 | 国产香蕉视频 | a级片在线| 精品欧美一区二区精品久久久 |