iOS開發(fā)-電子圍欄區(qū)域監(jiān)聽深入篇

1.前言:

  • 這篇的主題寫的不是基礎實現(xiàn),如果想看入門篇可以看下面的文章:
    iOS地圖 -- 區(qū)域監(jiān)聽的實現(xiàn)和小練習
    Core Location 電子圍欄:入門
  • 當然,如果你集成的是三方框架,比如百度地圖和高德地圖,那你就照著官方文檔來.
  • 這篇主要是記錄我在實踐的過程中遇到的一些疑問以及解決的過程.這其中的點是網(wǎng)上一些入門文章沒有提到.所以一方面是對自己的總結方便以后溫故而知新,另一方面也希望可以幫到一些剛接觸這方面的人

2.關于位置訪問權限的問題:

  • 電子圍欄功能需要用戶同意"始終訪問"這一項,"僅使用期間""拒絕訪問"這兩個權限都會使該功能不能達到預期的效果,會有問題.拒絕狀態(tài)直接導致該功能無法使用.僅使用期間會導致App退到后臺或者在控制中心手動殺掉后不能正常使用.
  • 所以,每次使用該功能前,最好獲取一下用戶的權限設置,如果是拒絕狀態(tài)可以提示用戶,并引導跳轉到權限設置界面.如果是僅使用期間狀態(tài),可以給與用戶提示切換成始終訪問權限.

3.關于CLRegion:

請使用CLRegion的子類,比如:CLCircularRegion.

4.核心代碼(swift為例):

// 創(chuàng)建監(jiān)聽區(qū)域
let region = CLCircularRegion.init(center: coordinate, radius: distance, identifier: type.rawValue)
// 開始監(jiān)聽
self.locationManager.startMonitoring(for: region)
// 延時2秒后獲取圍欄狀態(tài)(為什么延時,請看后文)
DispatchQueue.main.asyncAfter(deadline: .now() + 2) {
  self.locationManager.requestState(for: region)
}
  • 調(diào)用startMonitoring開始監(jiān)聽.設置代理后主要關注以下代理回調(diào):
    func locationManager(_ manager: CLLocationManager, didEnterRegion region: CLRegion) {
        print("...進入電子圍欄...")
    }
    
    
    func locationManager(_ manager: CLLocationManager, didExitRegion region: CLRegion) {
        print("...離開電子圍欄...")
    }
    
    
    func locationManager(_ manager: CLLocationManager, monitoringDidFailFor region: CLRegion?, withError error: Error) {
        print("...監(jiān)聽圍欄失敗:\(region!), error:\(error)")
    }
    
    
    func locationManager(_ manager: CLLocationManager, didStartMonitoringFor region: CLRegion) {
        print("...開始監(jiān)聽...")
    }
  • 調(diào)用requestState方法可以獲取到當前位置狀態(tài),設置代理后會執(zhí)行以下代理回調(diào):
    func locationManager(_ manager: CLLocationManager, didDetermineState state: CLRegionState, for region: CLRegion) {
        let regionType = RegionType.init(rawValue: region.identifier)
        guard regionType != .unknown else {return}
        let str = state == .inside ? "狀態(tài): 電子圍欄內(nèi)" : (state == .outside ? "狀態(tài): 電子圍欄外" : "狀態(tài): 未知")
        if regionType == .company
        {
            self.clabel.text = str
        }
        else
        {
            self.hLabel.text = str
        }
    }

5.關于監(jiān)聽電子圍欄數(shù)量問題:

  • 一個App最多只能同時監(jiān)聽20個電子圍欄,超過后會走monitoringDidFail回調(diào).會報(Domain=kCLErrorDomain Code=5)的錯誤提示.

6.關于startMonitoring方法和requestState方法的區(qū)別,以及為何要requestState延時調(diào)用問題

  • 對于startMonitoring和requestState的區(qū)別,這里是我自己的一些理解,可能對可能不對,僅供參考.
  • startMonitoring調(diào)用后,即就開始了圍欄的監(jiān)聽,只要沒有移除監(jiān)聽,一旦狀態(tài)發(fā)生變化,就會走對應的代理回調(diào)方法.
  • requestState看官方注釋不難理解,異步的獲取當前電子圍欄的狀態(tài)(是否在電子圍欄內(nèi),是否在電子圍欄外和未知狀態(tài)).
  • 一般我們使用startMonitoring開啟區(qū)域監(jiān)聽后,都會調(diào)用requestState來獲取一下初始狀態(tài).為何延時是因為如果立即調(diào)用的話,會有概率報(Domain=kCLErrorDomain Code=5)的錯誤提,導致獲取失敗.

7.針對在始終訪問權限下App被銷毀后,關于移除電子圍欄你需要注意的問題

  • 首先提兩個問題,如果我監(jiān)聽了某個區(qū)域,然后在控制中心銷毀了App.請問此時,這個區(qū)域監(jiān)聽的功能還生效嗎?下一次進入App的時候,是否需要重新監(jiān)聽.
  • 針對第一個問題.通過代碼測試后,我得到了以下結果.當處于"始終訪問位置"權限時,只要沒有通過代碼來移除監(jiān)聽,即使App被銷毀了.系統(tǒng)還是會繼續(xù)處于監(jiān)聽狀態(tài).這個通過手機屏幕狀態(tài)欄左上角位置訪問小角標并沒有消失就可以確定
    880DC4D0F5DD8C41F766C1D797404985.png
  • 針對第二個問題:CLLocationManager有一個可以獲取當前監(jiān)聽了哪些電子圍欄的集合.


    4FC4D04F-3CBD-4D40-87BB-44DA132CA769.png

實踐:
1.當我沒有開始監(jiān)聽時,獲取此集合的count = 0;
2.當我開啟一個電子圍欄后,銷毀App.然后重啟App.獲取此集合的count = 1
結論:
App被銷毀后,下次重啟App.之前沒有移除的電子圍欄仍然處于監(jiān)聽狀態(tài).不需要重新添加.
所以:這里需要特別注意,前面提到了一個App最多只能監(jiān)聽20個區(qū)域.因此電子圍欄的監(jiān)聽和移除管理,自己要心里特別清晰.哪些不用了需要及時移除,否則會占用不必要的名額.另外一點就是,如果某個電子圍欄不需要監(jiān)聽了請及時移除,否則,即使用戶銷毀了App,仍然還是會占用系統(tǒng)資源,背地里在使用用戶的位置權限.作為強迫癥的我可受不了.

8.最后再說一下didEnterRegion和didExitRegion這兩個代理回調(diào)的執(zhí)行

  • 一個是進入電子圍欄會觸發(fā),一個是離開電子圍欄會觸發(fā)
  • app處于后臺,狀態(tài)發(fā)生改變了.回調(diào)是否會調(diào)用 --> 答案是會的
  • app被銷毀后,電子圍欄處于監(jiān)聽狀態(tài),狀態(tài)發(fā)生改變后,回調(diào)是否會被調(diào)用 --> 我之前心里想著是不會,因為App都被銷毀了,內(nèi)部代碼應該不會執(zhí)行吧.結果我通過注冊本地通知的方式來驗證后,答案是依然會執(zhí)行.
    代碼如下:
    func locationManager(_ manager: CLLocationManager, didEnterRegion region: CLRegion) {
        print("...進入電子圍欄...")
        self.locationManager.requestState(for: region)
        let type = RegionType.init(rawValue: region.identifier)!
        let str = type == .home ? "??" : "公司"
        self.addLocalNotification(body: "你已經(jīng)進入\(str)電子圍欄內(nèi)")
    }
    
    
    func locationManager(_ manager: CLLocationManager, didExitRegion region: CLRegion) {
        print("...離開電子圍欄...")
        self.locationManager.requestState(for: region)
        let type = RegionType.init(rawValue: region.identifier)!
        let str = type == .home ? "??" : "公司"
        self.addLocalNotification(body: "你已經(jīng)離開\(str)電子圍欄")
    }

代碼思路很簡單,我把注冊本地通知的代碼寫在了代理回調(diào)里.如果回調(diào)執(zhí)行了,那么本地通知就能注冊成功,我就能收到通知.如果不執(zhí)行,本地通知就不會被注冊,我就收不到通知.最后結果如圖:


0927B894-6F6A-4D3D-A22F-25E0D8A8ABB4.png

9.寫在最后

在驗證上文那些內(nèi)容的時候,我順帶寫了一個小項目,demo里弄了2個電子圍欄,一個是公司的一個是租房的.每天上下班可以監(jiān)聽我是否到公司了,是否到家了.無論是離開還是進入電子圍欄都會給我發(fā)個本地通知.然后就是當時Domain=kCLErrorDomain Code=5這個問題困住了我許久.解決的時候參考了以下文章(其實沒幫到我什么,但是code=5的原因很多,如果你也遇到了,也許這里會有你想要的):

最后附上demo地址:github

?著作權歸作者所有,轉載或內(nèi)容合作請聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時請結合常識與多方信息審慎甄別。
平臺聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點,簡書系信息發(fā)布平臺,僅提供信息存儲服務。

相關閱讀更多精彩內(nèi)容

  • 作為我十分喜愛的月華組為snk做的最后一款游戲,我給這個游戲的評價是SNK的巔峰之作。 不管對月華劍士系列的格斗手...
    小樓閱讀 747評論 0 2
  • 我要為《戰(zhàn)狼2》票房破50億,成為中國電影第一個進入全球票房前100,喝彩; 我要為紀錄片二十二票房破1億,成為中...
    王曉蘇閱讀 619評論 4 8
  • 總有一些文字在心里潮濕 歲月遷韶華,昨日青蔥,轉眼已有白發(fā)。 仿佛就是在昨天,自己還背著書包匆匆趕往家中,趕著看6...
    知更喵閱讀 421評論 0 1
  • 今天因為辦公室就剩下我一個人閑著,所以很幸運的參加了小組長以上級別的會議!參加了這種會議后,我深刻感受到自己處于...
    繁華落盡是孤獨閱讀 335評論 0 0

友情鏈接更多精彩內(nèi)容