重拾百度定位之踩坑篇(上)
前言
最近更新項(xiàng)目中用的百度定位SDK時(shí)遇見了一個(gè)奇葩的問題。當(dāng)升級(jí)SDK后百度定位一直返回505,通過百度定位官網(wǎng)查看該碼表示AK非法或者不存在。很糾結(jié),于是自己又寫了一個(gè)demo來研究一下百度定位以及大家使用百度定位經(jīng)常出現(xiàn)的問題,特此記錄。這篇文章我先將百度定位的實(shí)現(xiàn)也介紹一下,最后再分析遇到的問題及解決方案。
定位分析
目前百度定位提供了WIFI,基站,GPS等多種定位方式,適用于室內(nèi)、室外多種定位場景,具有出色的定位性能:定位精度高(其實(shí)我是想吐槽的)、覆蓋率廣、網(wǎng)絡(luò)定位請(qǐng)求流量小、定位速度快。
集成定位SDK
現(xiàn)在官網(wǎng)提供的最新的定位SDK版本是v7.0,官網(wǎng)SDK下載地址請(qǐng)戳 定位SDK,可根據(jù)自己的需要下載,在這里我進(jìn)入全部下載,只下載了全量定位。在新版本V7.0中百度將定位對(duì)開發(fā)包實(shí)現(xiàn)了分離
(1)基礎(chǔ)定位:開發(fā)包體積最小,但只包含基礎(chǔ)定位能力(GPS/WiFi/基站)、基礎(chǔ)位置描述能力;
(2)離線定位:在基礎(chǔ)定位能力基礎(chǔ)之上,提供離線定位能力,可在網(wǎng)絡(luò)環(huán)境不佳時(shí),進(jìn)行精準(zhǔn)定位;
(3)室內(nèi)定位:在基礎(chǔ)定位能力基礎(chǔ)之上,提供室內(nèi)高精度定位能力,精度可達(dá)1-3米;
(4)全量定位:包含離線定位、室內(nèi)高精度定位能力,同時(shí)提供更人性化的位置描述服務(wù);
對(duì)于這四種類型定位開發(fā)包是互斥的,一個(gè)應(yīng)用中只需集成一種定位開發(fā)包即可。下載成功之后,將jar包和.so文件放到對(duì)應(yīng)的文件下即可。
申請(qǐng)秘鑰
使用百度定位,我們需要在官網(wǎng)申請(qǐng)一個(gè)AK,項(xiàng)目定位時(shí)需要使用這個(gè)Ak,一個(gè)應(yīng)用對(duì)于一個(gè)AK,AK申請(qǐng)時(shí)需要提供包名及SHA1值。具體方式
可去官網(wǎng)查看。在這里我簡單介紹下SHA1獲取方式。在申請(qǐng)Ak時(shí),頁面填寫發(fā)布版SHA1和開發(fā)版SHA1。下面我提供兩種方式獲取SHA1值。
AndroidStudio Terminal獲取
- -rfc 以 RFC 樣式輸出
- -alias <alias> 要處理的條目的別名
- -keystore <keystore> 密鑰庫名稱
- -storepass <arg> 密鑰庫口令
- -storetype <storetype> 密鑰庫類型
- -providername <providername> 提供方名稱
- -providerclass <providerclass> 提供方類名
- -providerarg <arg> 提供方參數(shù)
- -providerpath <pathlist> 提供方類路徑
- -v 詳細(xì)輸出
- -protected 通過受保護(hù)的機(jī)制的口令
上面是獲取密鑰庫信息的一些命令,則在此獲取SHA1可以
- keytool -v -list -keystore 【密鑰庫文件路徑】 -storepass 【密鑰庫文件密碼】
在Terminal執(zhí)行命令后就出現(xiàn)上面的詳細(xì)信息。SHA1后面的那一串字符就是我們需要的SHA1.
CMD方式
如果要在CMD中獲取,必須先要設(shè)置環(huán)境變量,具體設(shè)置方式可谷歌搜索。當(dāng)然獲取的命令和在AndroidStudio中獲取是一樣的。在上面我獲取下開發(fā)版SHA1。對(duì)于debug版一般存用戶下的.android目錄下,我們打開CMD后執(zhí)行 cd .android然后通過dir就可以看到目錄下會(huì)有一個(gè)debug.keystore文件,我們找的就是它。
在圖中你會(huì)看到?jīng)]有寫-storepass參數(shù)(當(dāng)然也可和上面一樣)。在回車后會(huì)提示輸入密鑰庫口令,對(duì)于我們的debug版本口令默認(rèn)是android,輸入后回車即可看到詳細(xì)信息了。
環(huán)境配置
要想實(shí)現(xiàn)定位,我們必須在清單文件中加入一些必要的權(quán)限以及key等信息,如下
- <!--百度定位權(quán)限相關(guān)-->
- <!-- 這個(gè)權(quán)限用于進(jìn)行網(wǎng)絡(luò)定位-->
- <uses-permission android:name="android.permission.ACCESS_COARSE_LOCATION"></uses-permission>
- <!-- 這個(gè)權(quán)限用于訪問GPS定位-->
- <uses-permission android:name="android.permission.ACCESS_FINE_LOCATION"></uses-permission>
- <!-- 用于訪問wifi網(wǎng)絡(luò)信息,wifi信息會(huì)用于進(jìn)行網(wǎng)絡(luò)定位-->
- <uses-permission android:name="android.permission.ACCESS_WIFI_STATE"></uses-permission>
- <!-- 獲取運(yùn)營商信息,用于支持提供運(yùn)營商信息相關(guān)的接口-->
- <uses-permission android:name="android.permission.ACCESS_NETWORK_STATE"></uses-permission>
- <!-- 這個(gè)權(quán)限用于獲取wifi的獲取權(quán)限,wifi信息會(huì)用來進(jìn)行網(wǎng)絡(luò)定位-->
- <uses-permission android:name="android.permission.CHANGE_WIFI_STATE"></uses-permission>
- <!-- 用于讀取手機(jī)當(dāng)前的狀態(tài)-->
- <uses-permission android:name="android.permission.READ_PHONE_STATE"></uses-permission>
- <!-- 寫入擴(kuò)展存儲(chǔ),向擴(kuò)展卡寫入數(shù)據(jù),用于寫入離線定位數(shù)據(jù)-->
- <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"></uses-permission>
- <!-- 訪問網(wǎng)絡(luò),網(wǎng)絡(luò)定位需要上網(wǎng)-->
- <uses-permission android:name="android.permission.INTERNET" />
- <!-- SD卡讀取權(quán)限,用戶寫入離線定位數(shù)據(jù)-->
- <uses-permission android:name="android.permission.MOUNT_UNMOUNT_FILESYSTEMS"> </uses-permission>
- <application
- android:allowBackup="true"
- android:icon="@mipmap/ic_launcher"
- android:label="@string/app_name"
- android:supportsRtl="true"
- android:theme="@style/AppTheme">
- <service
- android:name="com.baidu.location.f"
- android:enabled="true"
- android:process=":remote" >
- </service>
- <meta-data
- android:name="com.baidu.lbsapi.API_KEY"
- android:value="w7NQOKL8SpxHrs6lixBNoe90" />
- </application>
定位實(shí)現(xiàn)
對(duì)于定位的實(shí)現(xiàn)我們可以分為三步,第一步:初始化LocationClient;第二步:通過LocationClientOption設(shè)置定位參數(shù);第三步:實(shí)現(xiàn)BDLocationListener接口??粗遣皇呛芎唵?,你沒看錯(cuò),確實(shí)很簡單。
初始化LocationClient
- /**
- * 獲取LocationService實(shí)例
- *
- * @param context
- * @return
- */
- public static LocationService getInstance(Context context) {
- if (locationClient == null) {
- synchronized (LocationService.class) {
- locationService= new LocationService(context);
- }
- }
- return locationService;
- }
- private LocationService(Context context) {
- if (locationClient == null) {
- locationClient = new LocationClient(context);
- locationClient.setLocOption(getDefaultLocationClientOption());
- }
- }
設(shè)置定位參數(shù)
- /***
- * 配置參數(shù)
- *
- * @return DefaultLocationClientOption
- */
- public LocationClientOption getDefaultLocationClientOption() {
- if (locationClientOption == null) {
- locationClientOption = new LocationClientOption();
- locationClientOption.setLocationMode(LocationClientOption.LocationMode.Hight_Accuracy);//可選,默認(rèn)高精度,設(shè)置定位模式,高精度,低功耗,僅設(shè)備
- locationClientOption.setCoorType("bd09ll");//可選,默認(rèn)gcj02,設(shè)置返回的定位結(jié)果坐標(biāo)系,如果配合百度地圖使用,建議設(shè)置為bd09ll;
- locationClientOption.setScanSpan(3000);//可選,默認(rèn)0,即僅定位一次,設(shè)置發(fā)起定位請(qǐng)求的間隔需要大于等于1000ms才是有效的
- locationClientOption.setIsNeedAddress(true);//可選,設(shè)置是否需要地址信息,默認(rèn)不需要
- locationClientOption.setIsNeedLocationDescribe(true);//可選,設(shè)置是否需要地址描述
- locationClientOption.setNeedDeviceDirect(true);//可選,設(shè)置是否需要設(shè)備方向結(jié)果
- locationClientOption.setLocationNotify(true);//可選,默認(rèn)false,設(shè)置是否當(dāng)gps有效時(shí)按照1S1次頻率輸出GPS結(jié)果
- locationClientOption.setIgnoreKillProcess(true);//可選,默認(rèn)true,定位SDK內(nèi)部是一個(gè)SERVICE,并放到了獨(dú)立進(jìn)程,設(shè)置是否在stop的時(shí)候殺死這個(gè)進(jìn)程,默認(rèn)不殺死
- locationClientOption.setIsNeedLocationDescribe(true);//可選,默認(rèn)false,設(shè)置是否需要位置語義化結(jié)果,可以在BDLocation.getLocationDescribe里得到,結(jié)果類似于“在北京天安門附近”
- locationClientOption.setIsNeedLocationPoiList(true);//可選,默認(rèn)false,設(shè)置是否需要POI結(jié)果,可以在BDLocation.getPoiList里得到
- locationClientOption.SetIgnoreCacheException(false);//可選,默認(rèn)false,設(shè)置是否收集CRASH信息,默認(rèn)收集
- locationClientOption.setIsNeedAltitude(false);//可選,默認(rèn)false,設(shè)置定位時(shí)是否需要海拔信息,默認(rèn)不需要,除基礎(chǔ)定位版本都可用
- }
- return locationClientOption;
- }
接下文