Android

安卓百度地图(一)定位功能的实现、周边POI的检索

2018-01-11  本文已影响5124人  Active_Loser

其他文章:
安卓百度离线地图的下载以及使用
安卓百度地图(二)地图显示以及离线地图的下载使用
安卓百度地图(三)绘制点、线等图层信息
安卓百度地图(四)城市,周边,区域检索
安卓百度地图(五)百度地图路线规划
安卓百度地图(六)鹰眼轨迹的上传,历史轨迹的显示
安卓百度地图(七)地理围栏的建立与使用

本文主要针对百度定位sdk的api进行整理

一 定位简介

百度地图Android定位SDK提供GPS、基站、WiFi、地磁、蓝牙、传感器等多种定位方式,适用于室内、室外多种定位场景,具有出色的定位性能:定位精度高、覆盖率广、网络定位请求流量小、定位速度快。 引用官方的图片

在室内无gps时,百度定位就是利用的WiFi和基站定位的,在室外有gps时,百度是利用的android自带的LocationManager进行定位,当然室外也可以用WiFi基站定位。

二 百度地图配置

关于百度地图的配置,可以查看官网的详解教程:传送门

android {
    sourceSets {
        main {
            jniLibs.srcDirs = ['libs']
        }
    }
}
<meta-data
    android:name="com.baidu.lbsapi.API_KEY"
    android:value="开发者申请的AK" >
</meta-data>
//Application标签中声明service组件,每个App拥有自己单独的定位service
<service
     android:name="com.baidu.location.f" 
    android:enabled="true" 
    android:process=":remote"> 
</service>
<!-- 这个权限用于进行网络定位-->
<uses-permission android:name="android.permission.ACCESS_COARSE_LOCATION"></uses-permission>
<!-- 这个权限用于访问GPS定位-->
<uses-permission android:name="android.permission.ACCESS_FINE_LOCATION"></uses-permission>
<!-- 用于访问wifi网络信息,wifi信息会用于进行网络定位-->
<uses-permission android:name="android.permission.ACCESS_WIFI_STATE"></uses-permission>
<!-- 获取运营商信息,用于支持提供运营商信息相关的接口-->
<uses-permission android:name="android.permission.ACCESS_NETWORK_STATE"></uses-permission>
<!-- 这个权限用于获取wifi的获取权限,wifi信息会用来进行网络定位-->
<uses-permission android:name="android.permission.CHANGE_WIFI_STATE"></uses-permission>
<!-- 用于读取手机当前的状态-->
<uses-permission android:name="android.permission.READ_PHONE_STATE"></uses-permission>
<!-- 写入扩展存储,向扩展卡写入数据,用于写入离线定位数据-->
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"></uses-permission>
<!-- 访问网络,网络定位需要上网-->
<uses-permission android:name="android.permission.INTERNET" />
<!-- SD卡读取权限,用户写入离线定位数据-->
<uses-permission android:name="android.permission.MOUNT_UNMOUNT_FILESYSTEMS"></uses-permission>

注意:安卓6.0系统以上,一些权限需要动态申请

三 百度地图定位

1. 初始化LocationClient类

public LocationClient mLocationClient = null;;
public void onCreate()
{
     mLocationClient = new LocationClient(getApplicationContext());//声明LocationClient类
     mLocationClient.registerLocationListener(myListener);//注册监听函数
}

2. 配置定位SDK参数

LocationClientOption mOption = new LocationClientOption();

/**
* 默认高精度,设置定位模式
* LocationMode.Hight_Accuracy 高精度定位模式:这种定位模式下,会同时使用
* LocationMode.Battery_Saving 低功耗定位模式:这种定位模式下,不会使用GPS,只会使用网络定位。
* LocationMode.Device_Sensors 仅用设备定位模式:这种定位模式下,
*/
mOption.setLocationMode(LocationMode.Hight_Accuracy);

/**
* 默认是true,设置是否使用gps定位
* 如果设置为false,即使mOption.setLocationMode(LocationMode.Hight_Accuracy)也不会gps定位
*/
mOption.setOpenGps(true);

/**
* 默认gcj02,设置返回的定位结果坐标系,如果配合百度地图使用,建议设置为bd09ll;
* 目前国内主要有以下三种坐标系:
1. wgs84:目前广泛使用的GPS全球卫星定位系统使用的标准坐标系;
2. gcj02:经过国测局加密的坐标;
3. bd09:为百度坐标系,其中bd09ll表示百度经纬度坐标,bd09mc表示百度墨卡托米制坐标;
* 在国内获得的坐标系类型可以是:国测局坐标、百度墨卡托坐标 和 百度经纬度坐标。
  在海外地区,只能获得WGS84坐标。请在使用过程中注意选择坐标。
*/
mOption.setCoorType("bd09ll");

/**
* 默认0,即仅定位一次;设置间隔需大于等于1000ms,表示周期性定位
* 如果不在AndroidManifest.xml声明百度指定的Service,周期性请求无法正常工作
* 这里需要注意的是:如果是室外gps定位,不用访问服务器,设置的间隔是3秒,那么就是3秒返回一次位置
  如果是WiFi基站定位,需要访问服务器,这个时候每次网络请求时间差异很大,设置的间隔是3秒,
  只能大概保证3秒左右会返回就一次位置,有时某次定位可能会5秒才返回
*/
mOption.setScanSpan(3000);

/**
* 默认false,设置是否需要地址信息
* 返回省、市、区、街道等地址信息,这个api用处很大,
  很多新闻类app会根据定位返回的市区信息推送用户所在市的新闻
*/
mOption.setIsNeedAddress(true);

/**
* 默认false,设置是否需要位置语义化结果
* 可以在BDLocation.getLocationDescribe里得到,结果类似于“在北京天安门附近”
*/
mOption.setIsNeedLocationDescribe(true);

/**
* 默认false,设置是否需要设备方向传感器的方向结果
* 一般在室外gps定位时,返回的位置信息是带有方向的,但是有时候gps返回的位置也不带方向,
  这个时候可以获取设备方向传感器的方向
* wifi基站定位的位置信息是不带方向的,如果需要可以获取设备方向传感器的方向
*/
mOption.setNeedDeviceDirect(false);

/**
* 默认false,设置是否当gps有效时按照设定的周期频率输出GPS结果
* 室外gps有效时,周期性1秒返回一次位置信息,其实就是设置了
locationManager.requestLocationUpdates中的minTime参数为1000ms,1秒回调一个gps位置
* 如果设置了mOption.setScanSpan(3000),那minTime就是3000ms了,3秒回调一个gps位置
*/
mOption.setLocationNotify(false);

/**
* 默认true,定位SDK内部是一个SERVICE,并放到了独立进程,设置是否在stop的时候杀死这个进程,默认不杀死
* 如果你已经拿到了你要的位置信息,不需要再定位了,不杀死留着干嘛
*/
mOption.setIgnoreKillProcess(true);

/**
* 默认false,设置是否需要POI结果,可以在BDLocation.getPoiList里得到
* POI就是获取到的位置附近的一些商场、饭店、银行等信息
*/
mOption.setIsNeedLocationPoiList(true);

/**
* 默认false,设置是否收集CRASH信息,默认收集
*/
mOption.SetIgnoreCacheException(false);

/**
* 默认false,设置定位时是否需要海拔高度信息,默认不需要,除基础定位版本都可用
*/
mOption.setIsNeedAltitude(false);

mLocationClient.setLocOption(mOption);//设置定位参数

3. 发起定位

发起定位,便能够从BDAbstractLocationListener监听接口中获取定位结果信息。

//mLocationClient为第二步初始化过的LocationClient对象
//调用LocationClient的start()方法,便可发起定位请求
mLocationClient.start();
//tart():启动定位SDK;stop():关闭定位SDK。调用start()之后只需要等待定位结果自动回调即可。
//开发者定位场景如果是单次定位的场景,在收到定位结果之后直接调用stop()函数即可。
//如果stop()之后仍然想进行定位,可以再次start()等待定位结果回调即可。
//自v7.2版本起,新增LocationClient.reStart()方法,用于在某些特定的异常环境下重启定位。

4.实现BDAbstractLocationListener接口

发起定位后,即可获取所需要的信息

private class MyBDAbstractLocationListener extends BDAbstractLocationListener {
        @Override
        public void onReceiveLocation(BDLocation bdLocation) {
            //此处的BDLocation为定位结果信息类,通过它的各种get方法可获取定位相关的全部结果    
           if (null != location && location.getLocType() != BDLocation.TypeServerError) {
             
           }   
        }
}

4.1 获取基本定位信息

bdLocation.getLatitude();//纬度
bdLocation.getLongitude();//经度
bdLocation.getDirection();//定位方向
bdLocation.getRadius();//定位精度
bdLocation.getCoorType();//定位坐标类型
bdLocation.getLocType();//定位类型、定位错误返回码
bdLocation.getLocTypeDescription();//对应的定位类型说明
bdLocation.getTime();//获取经纬度服务器时间
//判断用户是在室内,还是在室外1:室内,0:室外,这个判断不一定是100%准确的
bdLocation.getUserIndoorState();

4.2 获取地址信息

注意:配置定位SDK参数中,添加获取地址信息option.setIsNeedAddress(true);

bdLocation.getAddrStr();    //获取详细地址信息
bdLocation.getCountry();    //获取国家
bdLocation.getProvince();    //获取省份
bdLocation.getCity();    //获取城市
bdLocation.getDistrict();    //获取区县
bdLocation.getStreet();    //获取街道信息

4.3 获取位置信息描述

注意:首先在配置定位SDK参数中,添加获取位置信息描述option.setIsNeedLocationDescribe(true);

location.getLocationDescribe();    //获取位置描述信息

4.4 周边POI信息

获取位置附近的一些商场、饭店、银行等信息
注意:首先在配置定位SDK参数中,添加获取周边信息option.setIsNeedLocationPoiList(true);
在实现的BDAbstractLocationListener接口中,通过getPoiList()方法,即可获取周边信息

List<Poi> poiList = bdLocation.getPoiList(); //POI信息包括POI ID、名称等,具体信息请参照类参考中POI类的相关说明

4.5 判断定位类型

if (bdLocation.getLocType() == BDLocation.TypeGpsLocation) {// GPS定位结果
        bdLocation.getSpeed();// 速度 单位:km/h
        bdLocation.getSatelliteNumber();// 卫星数目
        bdLocation.getAltitude();// 海拔高度 单位:米
        bdLocation.getGpsAccuracyStatus();// *****gps质量判断*****
        Toast.makeText(MainActivity.this, "gps定位成功", Toast.LENGTH_SHORT).show();

} else if (bdLocation.getLocType() == BDLocation.TypeNetWorkLocation) {// 网络定位结果
       // 运营商信息
        if (bdLocation.hasAltitude()) {// *****如果有海拔高度*****
            bdLocation.getAltitude();// 单位:米
        }
        (bdLocation.getOperators();   // 运营商信息
        Toast.makeText(MainActivity.this, "网络定位成功", Toast.LENGTH_SHORT).show();

} else if (bdLocation.getLocType() == BDLocation.TypeOffLineLocation) {// 离线定位结果
        Toast.makeText(MainActivity.this, "离线定位成功,离线定位结果也是有效的", Toast.LENGTH_SHORT).show();

} else if (bdLocation.getLocType() == BDLocation.TypeServerError) {
        Toast.makeText(MainActivity.this, "服务端网络定位失败,可以反馈IMEI号和大体定位时间到loc-bugs@baidu.com,会有人追查原因", Toast.LENGTH_SHORT).show();

} else if (bdLocation.getLocType() == BDLocation.TypeNetWorkException) {
        Toast.makeText(MainActivity.this, "网络不通导致定位失败,请检查网络是否通畅", Toast.LENGTH_SHORT).show();

} else if (bdLocation.getLocType() == BDLocation.TypeCriteriaException) {
              
        Toast.makeText(MainActivity.this, "法获取有效定位依据导致定位失败,一般是由于手机的原因,处于飞行模式下一般会造成这种结
果,可以试着重启手机", Toast.LENGTH_SHORT).show();
}

4.6 国内外位置判断

//此处的BDLocation为定位结果信息类,通过它的各种get方法可获取定位相关的全部结果
//以下只列举与国内外判断相关的内容
//更多结果信息获取说明,请参照类参考中BDLocation类中的说明
//BDLocation.getLocationWhere()方法可获得当前定位点是否是国内,它的取值及含义如下:
//BDLocation.LOCATION_WHERE_IN_CN:当前定位点在国内;
//BDLocation.LOCATION_WHERE_OUT_CN:当前定位点在海外;
//其他:无法判定。

4.7 位置提醒

定位SDK支持位置提醒功能,位置提醒最多提醒3次,3次过后将不再提醒。假如需要再次提醒、或者要修改提醒点坐标,都可通过函数SetNotifyLocation()来实现。

4.7.1 注册监听函数

 //注册监听函数
 mLocationClient.registerNotify(myListener);    

4.7.2 实现位置监听的回调

定义MyNotifyLister类,继承BDNotifyListener,实现位置监听的回调。
public class MyNotifyLister extends BDNotifyListener {
    public void onNotify(BDLocation mlocation, float distance){
        //已到达设置监听位置附近
    }           
}

4.7.3 实现设置位置消息提醒

调用BDNotifyListener的setNotifyLocation方法,实现设置位置消息提醒。
myListener.setNotifyLocation(40.0f, 116.0f, 3000, mLocationClient.getLocOption().getCoorType());
//设置位置提醒,四个参数分别是:纬度,经度,距离范围,坐标系类型(gcj02,gps,bd09,bd09ll)

4.7.4 启动定位

启动定位,SDK便会自动开启位置消息提醒的监听。
调用LocationClient的start()方法,启动定位。核心代码如下:
mLocationClient.start();
//mLocationClient为第二步初始化过的LocationClient对象
//调用LocationClient的start()方法,开启定位

4.7.5 取消监听

调用BDNotifyListener的removeNotifyEvent方法,实现取消位置监听。核心代码如下:
myListener.removeNotifyEvent(myListener);
//myListener为第二步中定义过的BDNotifyListener对象
//调用执行removeNotifyEvent方法,即可实现取消监听

5. 释放资源

mLocationClient.unregisterListener(myLocationListener); //注销掉监听
mLocationClient.stop(); //停止定位

本文主要做于收集整理笔记使用,关于文章的源码,请移步我的GitHub

上一篇下一篇

猜你喜欢

热点阅读