Android封裝高德地圖定位工具類Util的詳細(xì)步驟
前提
每次做的項(xiàng)目中或者維護(hù)公司之前舊項(xiàng)目的時(shí)候,都會(huì)用到通過(guò)定位來(lái)獲取經(jīng)緯度,我們都知道,Android官方也提供了獲取經(jīng)緯度的方法,但是不太好使,所以就用了高德地圖的API,不能每次用的時(shí)候都要寫一堆代碼,效率挺低的,于是就想著,封裝成一個(gè)工具類,方便調(diào)用,為以后的項(xiàng)目,不管是管理方面還是查找方面都簡(jiǎn)潔了不少。
第一步、去官網(wǎng)創(chuàng)建高德Key
官網(wǎng)地址:lbs.amap.com/product/loc…


帶*號(hào)的填完后,點(diǎn)擊提交,獲取SHA1值如下(通過(guò)代碼獲取的)或者通過(guò)命令行獲取,兩者都行。
Android studio代碼獲取SHA1值
調(diào)用 Log.e("-->打印sha1 ","${sha1(this)}")
fun sha1(context: Context): String {
try {
val info: PackageInfo = context.packageManager.getPackageInfo(
context.packageName, PackageManager.GET_SIGNATURES)
val cert: ByteArray = info.signatures.get(0).toByteArray()
val md: MessageDigest = MessageDigest.getInstance("SHA1")
val publicKey: ByteArray = md.digest(cert)
val hexString = StringBuffer()
for (i in publicKey.indices) {
val appendString = Integer.toHexString(0xFF and publicKey[i].toInt())
.toUpperCase(Locale.US)
if (appendString.length == 1) hexString.append("0")
hexString.append(appendString)
hexString.append(":")
}
val result = hexString.toString()
return result.substring(0, result.length - 1)
} catch (e: PackageManager.NameNotFoundException) {
e.printStackTrace()
} catch (e: NoSuchAlgorithmException) {
e.printStackTrace()
}
return ""
}

第二步 通過(guò)Gradle集成SDK(方便):
1、在Project的build.gradle文件中配置repositories,添加maven或jcenter倉(cāng)庫(kù)地址:
allprojects { repositories { jcenter() // 或者 mavenCentral() } }
2、在主工程的build.gradle文件配置dependencies
//定位 implementation 'com.amap.api:location:latest.integration'
第三步 配置參數(shù)
第1步,配置AndroidManifest.xml
請(qǐng)?jiān)赼pplication標(biāo)簽中聲明service組件,每個(gè)app擁有自己?jiǎn)为?dú)的定位service。
<!-- 定位需要的服務(wù) 使用2.0的定位需要加上這個(gè) -->
<service android:name="com.amap.api.location.APSService" >
</service>
第2步,聲明權(quán)限 如果項(xiàng)目中已有其中的權(quán)限,那就不用加了
<!--用于進(jìn)行網(wǎng)絡(luò)定位--> <uses-permission android:name="android.permission.ACCESS_COARSE_LOCATION"></uses-permission> <!--用于訪問(wèn)GPS定位--> <uses-permission android:name="android.permission.ACCESS_FINE_LOCATION"></uses-permission> <!--用于獲取運(yùn)營(yíng)商信息,用于支持提供運(yùn)營(yíng)商信息相關(guān)的接口--> <uses-permission android:name="android.permission.ACCESS_NETWORK_STATE"></uses-permission> <!--用于訪問(wèn)wifi網(wǎng)絡(luò)信息,wifi信息會(huì)用于進(jìn)行網(wǎng)絡(luò)定位--> <uses-permission android:name="android.permission.ACCESS_WIFI_STATE"></uses-permission> <!--用于獲取wifi的獲取權(quán)限,wifi信息會(huì)用來(lái)進(jìn)行網(wǎng)絡(luò)定位--> <uses-permission android:name="android.permission.CHANGE_WIFI_STATE"></uses-permission> <!--用于訪問(wèn)網(wǎng)絡(luò),網(wǎng)絡(luò)定位需要上網(wǎng)--> <uses-permission android:name="android.permission.INTERNET"></uses-permission> <!--用于讀取手機(jī)當(dāng)前的狀態(tài)--> <uses-permission android:name="android.permission.READ_PHONE_STATE"></uses-permission> <!--用于寫入緩存數(shù)據(jù)到擴(kuò)展存儲(chǔ)卡--> <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"></uses-permission> <!--用于申請(qǐng)調(diào)用A-GPS模塊--> <uses-permission android:name="android.permission.ACCESS_LOCATION_EXTRA_COMMANDS"></uses-permission>
第3步,設(shè)置高德Key:
<meta-data android:name="com.amap.api.v2.apikey" android:value="key">//開(kāi)發(fā)者申請(qǐng)的key </meta-data>
第四步 獲取定位數(shù)據(jù)
在需要獲取經(jīng)緯度的頁(yè)面調(diào)用如下代碼:
private var amapLocationUtil: AmapLocationUtil? = null
fun initLocationOption() {
if (null == amapLocationUtil) {
amapLocationUtil = AmapLocationUtil(CommApplication.getApplication())
}
amapLocationUtil!!.initLocation()
amapLocationUtil!!.startLocation()
amapLocationUtil!!.setOnCallBackListener { longitude, latitude, location, isSucdess, address ->
//Log.e("--->", "longitude" + longitude + "\n" + "latitude" + latitude + "\n" + "isSucdess" + isSucdess + "\n" + "address" + address);
//Log.e("--->",location.getProvince()+ "\n" +location.getCity()+"\n"+location.getDistrict());
//isSucdess true 定位成功 false 失敗
if (isSucdess) {
} else {
//定位失敗,重試定位
amapLocationUtil!!.startLocation()
}
}
}
注意:如果是在當(dāng)前Activity實(shí)例化的,不要忘了銷毀
override fun onDestroy() {
super.onDestroy()
if (amapLocationUtil != null) {
amapLocationUtil!!.destroyLocation()
}
}
看到了代碼里調(diào)用了AmapLocationUtil,這個(gè)是我封裝好的一個(gè)工具類,方便調(diào)用,代碼如下:
/**
*
* Created by JasonYin
* Description:封裝高德地圖Util
*
*/
class AmapLocationUtil(private val mContext: Context) {
private var locationClient: AMapLocationClient? = null
private var locationOption: AMapLocationClientOption? = null
private var mOnCallBackListener: onCallBackListener? = null
/**
* 初始化定位
*/
fun initLocation() { //初始化client
if (null == locationClient) {
locationClient = AMapLocationClient(mContext)
}
locationOption = defaultOption
//設(shè)置定位參數(shù)
locationClient!!.setLocationOption(locationOption)
// 設(shè)置定位監(jiān)聽(tīng)
locationClient!!.setLocationListener(locationListener)
}//可選,設(shè)置定位模式,可選的模式有高精度、僅設(shè)備、僅網(wǎng)絡(luò)。默認(rèn)為高精度模式
//可選,設(shè)置是否gps優(yōu)先,只在高精度模式下有效。默認(rèn)關(guān)閉
//可選,設(shè)置網(wǎng)絡(luò)請(qǐng)求超時(shí)時(shí)間。默認(rèn)為30秒。在僅設(shè)備模式下無(wú)效
//可選,設(shè)置定位間隔。默認(rèn)為2秒
//可選,設(shè)置是否返回逆地理地址信息。默認(rèn)是true
//可選,設(shè)置是否單次定位。默認(rèn)是false
//可選,設(shè)置是否等待wifi刷新,默認(rèn)為false.如果設(shè)置為true,會(huì)自動(dòng)變?yōu)閱未味ㄎ?,持續(xù)定位時(shí)不要使用
//可選, 設(shè)置網(wǎng)絡(luò)請(qǐng)求的協(xié)議。可選HTTP或者HTTPS。默認(rèn)為HTTP
//可選,設(shè)置是否使用傳感器。默認(rèn)是false
//可選,設(shè)置是否開(kāi)啟wifi掃描。默認(rèn)為true,如果設(shè)置為false會(huì)同時(shí)停止主動(dòng)刷新,停止以后完全依賴于系統(tǒng)刷新,定位位置可能存在誤差
//可選,設(shè)置是否使用緩存定位,默認(rèn)為true
//可選,設(shè)置定位模式,可選的模式有高精度、僅設(shè)備、僅網(wǎng)絡(luò)。默認(rèn)為高精度模式
//可選,設(shè)置是否gps優(yōu)先,只在高精度模式下有效。默認(rèn)關(guān)閉
//如果網(wǎng)絡(luò)可用就選擇高精度
private val defaultOption: AMapLocationClientOption
private get() {
val mOption = AMapLocationClientOption()
//如果網(wǎng)絡(luò)可用就選擇高精度
if (NetworkUtils.isConnected()) { //可選,設(shè)置定位模式,可選的模式有高精度、僅設(shè)備、僅網(wǎng)絡(luò)。默認(rèn)為高精度模式
mOption.locationMode = AMapLocationClientOption.AMapLocationMode.Hight_Accuracy
mOption.isGpsFirst = true //可選,設(shè)置是否gps優(yōu)先,只在高精度模式下有效。默認(rèn)關(guān)閉
} else {
mOption.locationMode = AMapLocationClientOption.AMapLocationMode.Device_Sensors //可選,設(shè)置定位模式,可選的模式有高精度、僅設(shè)備、僅網(wǎng)絡(luò)。默認(rèn)為高精度模式
mOption.isGpsFirst = true //可選,設(shè)置是否gps優(yōu)先,只在高精度模式下有效。默認(rèn)關(guān)閉
}
mOption.httpTimeOut = 30000 //可選,設(shè)置網(wǎng)絡(luò)請(qǐng)求超時(shí)時(shí)間。默認(rèn)為30秒。在僅設(shè)備模式下無(wú)效
mOption.interval = 2000 //可選,設(shè)置定位間隔。默認(rèn)為2秒
mOption.isNeedAddress = true //可選,設(shè)置是否返回逆地理地址信息。默認(rèn)是true
mOption.isOnceLocation = false //可選,設(shè)置是否單次定位。默認(rèn)是false
mOption.isOnceLocationLatest = false //可選,設(shè)置是否等待wifi刷新,默認(rèn)為false.如果設(shè)置為true,會(huì)自動(dòng)變?yōu)閱未味ㄎ?,持續(xù)定位時(shí)不要使用
AMapLocationClientOption.setLocationProtocol(AMapLocationClientOption.AMapLocationProtocol.HTTP) //可選, 設(shè)置網(wǎng)絡(luò)請(qǐng)求的協(xié)議。可選HTTP或者HTTPS。默認(rèn)為HTTP
mOption.isSensorEnable = true //可選,設(shè)置是否使用傳感器。默認(rèn)是false
mOption.isWifiScan = true //可選,設(shè)置是否開(kāi)啟wifi掃描。默認(rèn)為true,如果設(shè)置為false會(huì)同時(shí)停止主動(dòng)刷新,停止以后完全依賴于系統(tǒng)刷新,定位位置可能存在誤差
mOption.isLocationCacheEnable = true //可選,設(shè)置是否使用緩存定位,默認(rèn)為true
return mOption
}
var locationListener = AMapLocationListener { location ->
val sb = StringBuilder()
if (null != location) { //errCode等于0代表定位成功,其他的為定位失敗,具體的可以參照官網(wǎng)定位錯(cuò)誤碼說(shuō)明
if (location.errorCode == 0) {
longitude = location.longitude
latitude = location.latitude
val district = location.district
locationSuccess(longitude, latitude, true, location, district)
//定位成功,停止定位:如果實(shí)時(shí)定位,就把stopLocation()關(guān)閉
stopLocation()
} else { //定位失敗
// sb.append("定位失敗" + "\n");
// sb.append("錯(cuò)誤碼:" + location.getErrorCode() + "\n");
// sb.append("錯(cuò)誤信息:" + location.getErrorInfo() + "\n");
// sb.append("錯(cuò)誤描述:" + location.getLocationDetail() + "\n");
// Log.e("---> 定位失敗", sb.toString());
LocationFarile(false, location)
}
} else {
LocationFarile(false, location)
}
}
private fun LocationFarile(isSucdess: Boolean, location: AMapLocation) {
if (mOnCallBackListener != null) {
mOnCallBackListener!!.onCallBack(0.0, 0.0, location, false, "")
}
}
fun locationSuccess(longitude: Double, latitude: Double, isSucdess: Boolean, location: AMapLocation?, address: String?) {
if (mOnCallBackListener != null) {
mOnCallBackListener!!.onCallBack(longitude, latitude, location, true, address)
}
}
fun setOnCallBackListener(listener: onCallBackListener?) {
mOnCallBackListener = listener
}
interface onCallBackListener {
fun onCallBack(longitude: Double, latitude: Double, location: AMapLocation?, isSucdess: Boolean, address: String?)
}
/**
* 開(kāi)始定位
*/
fun startLocation() {
locationClient!!.startLocation()
}
/**
* 停止定位
*/
fun stopLocation() {
locationClient!!.stopLocation()
}
/**
* 銷毀定位
*/
fun destroyLocation() {
if (null != locationClient) {
/**
* 如果AMapLocationClient是在當(dāng)前Activity實(shí)例化的,
* 在Activity的onDestroy中一定要執(zhí)行AMapLocationClient的onDestroy
*/
locationClient!!.onDestroy()
locationClient = null
locationOption = null
}
}
companion object {
var longitude = 0.0
var latitude = 0.0
}
}
總結(jié)
到此這篇關(guān)于Android封裝高德地圖定位工具類Util的文章就介紹到這了,更多相關(guān)Android封裝高德定位工具類內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Android手勢(shì)密碼實(shí)現(xiàn)實(shí)例代碼
本篇文章主要介紹了Android手勢(shì)密碼實(shí)現(xiàn)實(shí)例代碼,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2017-04-04
Android自定義密碼輸入框的簡(jiǎn)單實(shí)現(xiàn)過(guò)程
在最近的項(xiàng)目中,用戶需要輸入密碼,不想用系統(tǒng)鍵盤,就寫了一個(gè)自定義鍵盤,下面這篇文章主要給大家介紹了關(guān)于Android自定義密碼輸入框的簡(jiǎn)單實(shí)現(xiàn)過(guò)程,需要的朋友可以參考下2021-11-11
Android自定義流式布局的實(shí)現(xiàn)示例
這篇文章主要介紹了Android自定義流式布局的實(shí)現(xiàn)示例,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2020-12-12
Android自定義DigitalClock控件實(shí)現(xiàn)商品倒計(jì)時(shí)
這篇文章主要為大家詳細(xì)介紹了Android DigitalClock實(shí)現(xiàn)商品倒計(jì)時(shí),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2018-02-02
Android判斷手機(jī)是否聯(lián)網(wǎng)及自動(dòng)跳轉(zhuǎn)功能(收藏版)
這篇文章主要介紹了Android判斷手機(jī)是否聯(lián)網(wǎng)及自動(dòng)跳轉(zhuǎn)功能(收藏版),在一些手機(jī)端連接wifi我們經(jīng)常會(huì)遇到這樣的功能,今天小編通過(guò)實(shí)例截圖給大家介紹下,需要的朋友可以參考下2019-11-11
Android開(kāi)發(fā)實(shí)現(xiàn)瀏覽器全屏顯示功能
這篇文章主要介紹了Android開(kāi)發(fā)實(shí)現(xiàn)瀏覽器全屏顯示功能,涉及Android布局修改及相關(guān)屬性動(dòng)態(tài)設(shè)置操作技巧,需要的朋友可以參考下2017-09-09
Android App中使用LinearLayout進(jìn)行居中布局的實(shí)例講解
這篇文章主要介紹了Android App中使用LinearLayout進(jìn)行居中布局的實(shí)例講解,文中分別介紹了水平居中和垂直居中的相關(guān)線性布局,需要的朋友可以參考下2016-04-04

