1.前言:
- 這篇的主題寫的不是基礎實現,如果想看入門篇可以看下面的文章:
iOS地圖 -- 區域監聽的實現和小練習
Core Location 電子圍欄:入門 - 當然,如果你集成的是三方框架,比如百度地圖和高德地圖,那你就照著官方文檔來.
- 這篇主要是記錄我在實踐的過程中遇到的一些疑問以及解決的過程.這其中的點是網上一些入門文章沒有提到.所以一方面是對自己的總結方便以后溫故而知新,另一方面也希望可以幫到一些剛接觸這方面的人
2.關于位置訪問權限的問題:
- 電子圍欄功能需要用戶同意"始終訪問"這一項,"僅使用期間"和"拒絕訪問"這兩個權限都會使該功能不能達到預期的效果,會有問題.拒絕狀態直接導致該功能無法使用.僅使用期間會導致App退到后臺或者在控制中心手動殺掉后不能正常使用.
- 所以,每次使用該功能前,最好獲取一下用戶的權限設置,如果是拒絕狀態可以提示用戶,并引導跳轉到權限設置界面.如果是僅使用期間狀態,可以給與用戶提示切換成始終訪問權限.
3.關于CLRegion:
請使用CLRegion的子類,比如:CLCircularRegion.
4.核心代碼(swift為例):
// 創建監聽區域
let region = CLCircularRegion.init(center: coordinate, radius: distance, identifier: type.rawValue)
// 開始監聽
self.locationManager.startMonitoring(for: region)
// 延時2秒后獲取圍欄狀態(為什么延時,請看后文)
DispatchQueue.main.asyncAfter(deadline: .now() + 2) {
self.locationManager.requestState(for: region)
}
- 調用startMonitoring開始監聽.設置代理后主要關注以下代理回調:
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("...監聽圍欄失敗:\(region!), error:\(error)")
}
func locationManager(_ manager: CLLocationManager, didStartMonitoringFor region: CLRegion) {
print("...開始監聽...")
}
- 調用requestState方法可以獲取到當前位置狀態,設置代理后會執行以下代理回調:
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 ? "狀態: 電子圍欄內" : (state == .outside ? "狀態: 電子圍欄外" : "狀態: 未知")
if regionType == .company
{
self.clabel.text = str
}
else
{
self.hLabel.text = str
}
}
5.關于監聽電子圍欄數量問題:
- 一個App最多只能同時監聽20個電子圍欄,超過后會走monitoringDidFail回調.會報(Domain=kCLErrorDomain Code=5)的錯誤提示.
6.關于startMonitoring方法和requestState方法的區別,以及為何要requestState延時調用問題
- 對于startMonitoring和requestState的區別,這里是我自己的一些理解,可能對可能不對,僅供參考.
- startMonitoring調用后,即就開始了圍欄的監聽,只要沒有移除監聽,一旦狀態發生變化,就會走對應的代理回調方法.
- requestState看官方注釋不難理解,異步的獲取當前電子圍欄的狀態(是否在電子圍欄內,是否在電子圍欄外和未知狀態).
- 一般我們使用startMonitoring開啟區域監聽后,都會調用requestState來獲取一下初始狀態.為何延時是因為如果立即調用的話,會有概率報(Domain=kCLErrorDomain Code=5)的錯誤提,導致獲取失敗.
7.針對在始終訪問權限下App被銷毀后,關于移除電子圍欄你需要注意的問題
- 首先提兩個問題,如果我監聽了某個區域,然后在控制中心銷毀了App.請問此時,這個區域監聽的功能還生效嗎?下一次進入App的時候,是否需要重新監聽.
-
針對第一個問題.通過代碼測試后,我得到了以下結果.當處于"始終訪問位置"權限時,只要沒有通過代碼來移除監聽,即使App被銷毀了.系統還是會繼續處于監聽狀態.這個通過手機屏幕狀態欄左上角位置訪問小角標并沒有消失就可以確定880DC4D0F5DD8C41F766C1D797404985.png
-
針對第二個問題:CLLocationManager有一個可以獲取當前監聽了哪些電子圍欄的集合.
4FC4D04F-3CBD-4D40-87BB-44DA132CA769.png
實踐:
1.當我沒有開始監聽時,獲取此集合的count = 0;
2.當我開啟一個電子圍欄后,銷毀App.然后重啟App.獲取此集合的count = 1
結論:
App被銷毀后,下次重啟App.之前沒有移除的電子圍欄仍然處于監聽狀態.不需要重新添加.
所以:這里需要特別注意,前面提到了一個App最多只能監聽20個區域.因此電子圍欄的監聽和移除管理,自己要心里特別清晰.哪些不用了需要及時移除,否則會占用不必要的名額.另外一點就是,如果某個電子圍欄不需要監聽了請及時移除,否則,即使用戶銷毀了App,仍然還是會占用系統資源,背地里在使用用戶的位置權限.作為強迫癥的我可受不了.
8.最后再說一下didEnterRegion和didExitRegion這兩個代理回調的執行
- 一個是進入電子圍欄會觸發,一個是離開電子圍欄會觸發
- app處于后臺,狀態發生改變了.回調是否會調用 --> 答案是會的
- app被銷毀后,電子圍欄處于監聽狀態,狀態發生改變后,回調是否會被調用 --> 我之前心里想著是不會,因為App都被銷毀了,內部代碼應該不會執行吧.結果我通過注冊本地通知的方式來驗證后,答案是依然會執行.
代碼如下:
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: "你已經進入\(str)電子圍欄內")
}
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: "你已經離開\(str)電子圍欄")
}
代碼思路很簡單,我把注冊本地通知的代碼寫在了代理回調里.如果回調執行了,那么本地通知就能注冊成功,我就能收到通知.如果不執行,本地通知就不會被注冊,我就收不到通知.最后結果如圖:
0927B894-6F6A-4D3D-A22F-25E0D8A8ABB4.png
9.寫在最后
在驗證上文那些內容的時候,我順帶寫了一個小項目,demo里弄了2個電子圍欄,一個是公司的一個是租房的.每天上下班可以監聽我是否到公司了,是否到家了.無論是離開還是進入電子圍欄都會給我發個本地通知.然后就是當時Domain=kCLErrorDomain Code=5這個問題困住了我許久.解決的時候參考了以下文章(其實沒幫到我什么,但是code=5的原因很多,如果你也遇到了,也許這里會有你想要的):
- https://github.com/evothings/phonegap-estimotebeacons/issues/94
- https://stackoverflow.com/questions/17733875/corelocation-kclerrordomain-error-5
最后附上demo地址:github