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í)行,本地通知就不會被注冊,我就收不到通知.最后結果如圖:

9.寫在最后
在驗證上文那些內(nèi)容的時候,我順帶寫了一個小項目,demo里弄了2個電子圍欄,一個是公司的一個是租房的.每天上下班可以監(jiān)聽我是否到公司了,是否到家了.無論是離開還是進入電子圍欄都會給我發(fā)個本地通知.然后就是當時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

