Android封装高德地图定位工具类Util

前提

每次做的项目中或者维护公司之前旧项目的时候,都会用到通过定位来获取经纬度,我们都知道,Android官方也提供了获取经纬度的方法,但是不太好使,所以就用了高德地图的API,不能每次用的时候都要写一堆代码,效率挺低的,于是就想着,封装成一个工具类,方便调用,为以后的项目,不管是管理方面还是查找方面都简洁了不少。

第一步、去官网创建高德Key

官网地址:lbs.amap.com/product/loc…

带*号的填完后,点击提交,获取SHA1值如下(通过代码获取的)或者通过命令行获取,两者都行。 Android studio代码获取SHA1值

调用 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 ""
    }
复制代码

第二步 通过Gradle集成SDK(方便):

1、在Project的build.gradle文件中配置repositories,添加maven或jcenter仓库地址:

allprojects { repositories { jcenter() // 或者 mavenCentral() } }
复制代码

2、在主工程的build.gradle文件配置dependencies

   //定位
   implementation 'com.amap.api:location:latest.integration'
复制代码

第三步 配置参数

第1步,配置AndroidManifest.xml

请在application标签中声明service组件,每个app拥有自己单独的定位service。

 <!-- 定位需要的服务 使用2.0的定位需要加上这个 -->
        <service android:name="com.amap.api.location.APSService" >
        </service>
复制代码

第2步,声明权限 如果项目中已有其中的权限,那就不用加了

<!--用于进行网络定位-->
<uses-permission android:name="android.permission.ACCESS_COARSE_LOCATION"></uses-permission> <!--用于访问GPS定位--> 
<uses-permission android:name="android.permission.ACCESS_FINE_LOCATION"></uses-permission>
<!--用于获取运营商信息,用于支持提供运营商信息相关的接口--> 
<uses-permission android:name="android.permission.ACCESS_NETWORK_STATE"></uses-permission>
<!--用于访问wifi网络信息,wifi信息会用于进行网络定位-->
<uses-permission android:name="android.permission.ACCESS_WIFI_STATE"></uses-permission>
<!--用于获取wifi的获取权限,wifi信息会用来进行网络定位-->
<uses-permission android:name="android.permission.CHANGE_WIFI_STATE"></uses-permission>
<!--用于访问网络,网络定位需要上网-->
<uses-permission android:name="android.permission.INTERNET"></uses-permission>
<!--用于读取手机当前的状态-->
<uses-permission android:name="android.permission.READ_PHONE_STATE"></uses-permission>
<!--用于写入缓存数据到扩展存储卡-->
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"></uses-permission> <!--用于申请调用A-GPS模块-->
<uses-permission android:name="android.permission.ACCESS_LOCATION_EXTRA_COMMANDS"></uses-permission>

复制代码

第3步,设置高德Key:

<meta-data android:name="com.amap.api.v2.apikey" android:value="key">//开发者申请的key           </meta-data>
复制代码

第四步 获取定位数据

在需要获取经纬度的页面调用如下代码:

    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()
            }
        }
    }

复制代码

注意:如果是在当前Activity实例化的,不要忘了销毁

override fun onDestroy() {
        super.onDestroy()
        if (amapLocationUtil != null) {
            amapLocationUtil!!.destroyLocation()
        }
    }
复制代码

看到了代码里调用了AmapLocationUtil,这个是我封装好的一个工具类,方便调用,代码如下:


/**
 *
 * 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
        //设置定位参数
        locationClient!!.setLocationOption(locationOption)
        // 设置定位监听
        locationClient!!.setLocationListener(locationListener)
    }//可选,设置定位模式,可选的模式有高精度、仅设备、仅网络。默认为高精度模式
    //可选,设置是否gps优先,只在高精度模式下有效。默认关闭
    //可选,设置网络请求超时时间。默认为30秒。在仅设备模式下无效
    //可选,设置定位间隔。默认为2秒
    //可选,设置是否返回逆地理地址信息。默认是true
    //可选,设置是否单次定位。默认是false
    //可选,设置是否等待wifi刷新,默认为false.如果设置为true,会自动变为单次定位,持续定位时不要使用
    //可选, 设置网络请求的协议。可选HTTP或者HTTPS。默认为HTTP
    //可选,设置是否使用传感器。默认是false
    //可选,设置是否开启wifi扫描。默认为true,如果设置为false会同时停止主动刷新,停止以后完全依赖于系统刷新,定位位置可能存在误差
    //可选,设置是否使用缓存定位,默认为true
//可选,设置定位模式,可选的模式有高精度、仅设备、仅网络。默认为高精度模式
    //可选,设置是否gps优先,只在高精度模式下有效。默认关闭

    //如果网络可用就选择高精度
    private val defaultOption: AMapLocationClientOption
        private get() {
            val mOption = AMapLocationClientOption()
            //如果网络可用就选择高精度
            if (NetworkUtils.isConnected()) { //可选,设置定位模式,可选的模式有高精度、仅设备、仅网络。默认为高精度模式
                mOption.locationMode = AMapLocationClientOption.AMapLocationMode.Hight_Accuracy
                mOption.isGpsFirst = true //可选,设置是否gps优先,只在高精度模式下有效。默认关闭
            } else {
                mOption.locationMode = AMapLocationClientOption.AMapLocationMode.Device_Sensors //可选,设置定位模式,可选的模式有高精度、仅设备、仅网络。默认为高精度模式
                mOption.isGpsFirst = true //可选,设置是否gps优先,只在高精度模式下有效。默认关闭
            }
            mOption.httpTimeOut = 30000 //可选,设置网络请求超时时间。默认为30秒。在仅设备模式下无效
            mOption.interval = 2000 //可选,设置定位间隔。默认为2秒
            mOption.isNeedAddress = true //可选,设置是否返回逆地理地址信息。默认是true
            mOption.isOnceLocation = false //可选,设置是否单次定位。默认是false
            mOption.isOnceLocationLatest = false //可选,设置是否等待wifi刷新,默认为false.如果设置为true,会自动变为单次定位,持续定位时不要使用
            AMapLocationClientOption.setLocationProtocol(AMapLocationClientOption.AMapLocationProtocol.HTTP) //可选, 设置网络请求的协议。可选HTTP或者HTTPS。默认为HTTP
            mOption.isSensorEnable = true //可选,设置是否使用传感器。默认是false
            mOption.isWifiScan = true //可选,设置是否开启wifi扫描。默认为true,如果设置为false会同时停止主动刷新,停止以后完全依赖于系统刷新,定位位置可能存在误差
            mOption.isLocationCacheEnable = true //可选,设置是否使用缓存定位,默认为true
            return mOption
        }

    var locationListener = AMapLocationListener { location ->
        val sb = StringBuilder()
        if (null != location) { //errCode等于0代表定位成功,其他的为定位失败,具体的可以参照官网定位错误码说明
            if (location.errorCode == 0) {
                longitude = location.longitude
                latitude = location.latitude
                val district = location.district
                locationSuccess(longitude, latitude, true, location, district)
                //定位成功,停止定位:如果实时定位,就把stopLocation()关闭
                stopLocation()
            } else { //定位失败
//                    sb.append("定位失败" + "\n");
//                    sb.append("错误码:" + location.getErrorCode() + "\n");
//                    sb.append("错误信息:" + location.getErrorInfo() + "\n");
//                    sb.append("错误描述:" + 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?)
    }

    /**
     * 开始定位
     */
    fun startLocation() {
        locationClient!!.startLocation()
    }

    /**
     * 停止定位
     */
    fun stopLocation() {
        locationClient!!.stopLocation()
    }

    /**
     * 销毁定位
     */
    fun destroyLocation() {
        if (null != locationClient) {
            /**
             * 如果AMapLocationClient是在当前Activity实例化的,
             * 在Activity的onDestroy中一定要执行AMapLocationClient的onDestroy
             */
            locationClient!!.onDestroy()
            locationClient = null
            locationOption = null
        }
    }

    companion object {
        var longitude = 0.0
        var latitude = 0.0
    }

}

本文在开源项目:https://github.com/Android-Alvin/Android-LearningNotes 中已收录,里面包含不同方向的自学编程路线、面试题集合/面经、及系列技术文章等,资源持续更新中...

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,864评论 6 494
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,175评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,401评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,170评论 1 286
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,276评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,364评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,401评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,179评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,604评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,902评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,070评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,751评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,380评论 3 319
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,077评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,312评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,924评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,957评论 2 351

推荐阅读更多精彩内容