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