首页 > 代码库 > Android studio 百度地图开发(2)地图定位

Android studio 百度地图开发(2)地图定位

Android studio 百度地图开发(2)地图定位

email:chentravelling@163.com

     开发环境:win7 64位,Android Studio,请注意是Android Studio,使用的百度地图定位SDK6.2.3

     地图显示、project配置请參考:Android studio 百度地图开发(1)配置project、显示地图

一.我为百度做点事

        最開始自己是从头到尾地写了一遍,可是始终不能成功定位。一直没找到原因。后来也是參考了百度官网上的Demo才弄出来。所以当中的代码,基本来自百度Demo。喜欢吐槽的朋友请键盘留情,然后悄悄飘过吧。

        一直以来都是喜欢百度的,虽然百度的的确确不能和Google比。可是。在中国的科技氛围里。我认为百度已经够好了。所以也贴一点百度地图定位SDK的产品优势。

        最后,我想问问那些嘴里骂着百度心里又离不开百度的朋友:你为中国的科技、IT、社会做了什么?

技术分享

二.定位SDK的应用

       言归正传。先贴整个project的代码,第三部分再解释说明。

(1)BaiDuMapActivity.java

package intvehapp.intvehapp;
import com.baidu.location.BDLocation;
import com.baidu.location.BDLocationListener;
import com.baidu.location.LocationClient;
import com.baidu.location.LocationClientOption;
import com.baidu.mapapi.SDKInitializer;
import com.baidu.mapapi.map.BaiduMap;
import com.baidu.mapapi.map.MapStatus;
import com.baidu.mapapi.map.MapStatusUpdateFactory;
import com.baidu.mapapi.map.MapView;
import com.baidu.mapapi.map.MyLocationData;
import com.baidu.mapapi.model.LatLng;

import android.app.Activity;
import android.os.Bundle;

public class BaiDuMapActivity extends Activity{
 /**
     * 定位SDK核心类
     */
    private LocationClient locationClient;
    /**
     * 定位监听
     */
    public MyLocationListenner myListener = new MyLocationListenner();
 /**
     * 百度地图控件
     */
    private MapView mapView;
    /**
     * 百度地图对象
     */
    private BaiduMap baiduMap;

    boolean isFirstLoc = true; // 是否首次定位

    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        SDKInitializer.initialize(getApplicationContext());
        setContentView(R.layout.activity_bai_du_map);
 /**
         * 地图初始化
         */
        //获取百度地图控件
        mapView = (MapView) findViewById(R.id.bmapView);
        //获取百度地图对象
        baiduMap = mapView.getMap();
        // 开启定位图层
        baiduMap.setMyLocationEnabled(true);
        /**
         * 定位初始化
         */
        //声明定位SDK核心类
        locationClient = new LocationClient(this);
        //注冊监听
        locationClient.registerLocationListener(myListener);
        //定位配置信息
        LocationClientOption option = new LocationClientOption();
        option.setOpenGps(true); // 打开gps
        option.setCoorType("bd09ll"); // 设置坐标类型
        option.setScanSpan(1000);//定位请求时间间隔
        locationClient.setLocOption(option);
        //开启定位
        locationClient.start();
    }

    /**
     * 定位SDK监听函数
     */
    public class MyLocationListenner implements BDLocationListener {

        @Override
        public void onReceiveLocation(BDLocation location) {
            // map view 销毁后不在处理新接收的位置
            if (location == null || mapView == null) {
                return;
            }
            MyLocationData locData = http://www.mamicode.com/new MyLocationData.Builder()>

(2)activity_bai_du_map.xml:添加百度地图控件

<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools" android:layout_width="match_parent"
    android:layout_height="match_parent" android:paddingLeft="@dimen/activity_horizontal_margin"
    android:paddingRight="@dimen/activity_horizontal_margin"
    android:paddingTop="@dimen/activity_vertical_margin"
    android:paddingBottom="@dimen/activity_vertical_margin"
    tools:context="intvehapp.intvehapp.BaiDuMapActivity">

    <com.baidu.mapapi.map.MapView
        android:id="@+id/bmapView"
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        android:clickable="true" />


</RelativeLayout>

(3)AndroidManifest.xml:添加百度地图SDK须要的权限、API_KEY和定位服务

<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
    package="intvehapp.intvehapp" >
    <!-- 百度地图API所需权限 -->
    <uses-permission android:name="android.permission.GET_ACCOUNTS" />
    <uses-permission android:name="android.permission.USE_CREDENTIALS" />
    <uses-permission android:name="android.permission.MANAGE_ACCOUNTS" />
    <uses-permission android:name="android.permission.AUTHENTICATE_ACCOUNTS" />
    <uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" />
    <uses-permission android:name="android.permission.INTERNET" />
    <uses-permission android:name="com.android.launcher.permission.READ_SETTINGS" />
    <uses-permission android:name="android.permission.CHANGE_WIFI_STATE" />
    <uses-permission android:name="android.permission.ACCESS_WIFI_STATE" />
    <uses-permission android:name="android.permission.READ_PHONE_STATE" />
    <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />
    <uses-permission android:name="android.permission.BROADCAST_STICKY" />
    <uses-permission android:name="android.permission.WRITE_SETTINGS" />
    <uses-permission android:name="android.permission.ACCESS_COARSE_LOCATION" />
    <!-- 訪问精确位置的权限 -->
    <uses-permission android:name="android.permission.ACCESS_FINE_LOCATION" />
    <application
        android:allowBackup="true"
        android:icon="@mipmap/ic_launcher"
        android:label="@string/app_name"
        android:supportsRtl="true"
        android:theme="@style/AppTheme" >
        <activity android:name=".BaiDuMapActivity" >
            <intent-filter>
                <action android:name="android.intent.action.MAIN" />

                <category android:name="android.intent.category.LAUNCHER" />
            </intent-filter>
        </activity>
        <!--百度API_KEY-->
        <meta-data
            android:name="com.baidu.lbsapi.API_KEY"
            android:value=http://www.mamicode.com/"iXDGdZtFiPifnNm6dUEFwDRXYQVeZ37V" />>
ok,通过这三部分,就能够成功定位了。效果例如以下:

技术分享

.定位SDK相关说明

【摘自百度官网

title=android-locsdk/guide/getloc">http://lbsyun.baidu.com/index.php?title=android-locsdk/guide/getloc】

【类參考:http://wiki.lbsyun.baidu.com/cms/androidloc/doc/v6_0_3/doc/index.html】

第一步,初始化LocationClient类

此处须要注意:LocationClient类必须在主线程中声明须要Context类型的參数

Context须要时全进程有效的context,推荐用getApplicationConext获取全进程有效的context

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

LocationClient类是定位SDK的核心类。详细方法详见类參考。

第二步,配置定位SDK參数

设置定位參数包含:定位模式(高精度定位模式。低功耗定位模式和仅用设备定位模式),返回坐标类型,是否打开GPS。是否返回地址信息、位置语义化信息、POI信息等等。

LocationClientOption类。该类用来设置定位SDK的定位方式,e.g.:

private void initLocation(){
        LocationClientOption option = new LocationClientOption();
        option.setLocationMode(LocationMode.Hight_Accuracy
);//可选,默认高精度,设置定位模式。高精度,低功耗,仅设备
        option.setCoorType("bd09ll");//可选。默认gcj02,设置返回的定位结果坐标系
        int span=1000;
        option.setScanSpan(span);//可选。默认0,即仅定位一次,设置发起定位请求的间隔须要大于等于1000ms才是有效的
        option.setIsNeedAddress(true);//可选。设置是否须要地址信息。默认不须要
        option.setOpenGps(true);//可选,默认false,设置是否使用gps
        option.setLocationNotify(true);//可选,默认false,设置是否当gps有效时依照1S1次频率输出GPS结果
        option.setIsNeedLocationDescribe(true);//可选。默认false。设置是否须要位置语义化结果。能够在BDLocation.getLocationDescribe里得到。结果相似于“在北京天安门附近”
        option.setIsNeedLocationPoiList(true);//可选。默认false。设置是否须要POI结果,能够在BDLocation.getPoiList里得到
option.setIgnoreKillProcess(false);//可选,默认true,定位SDK内部是一个SERVICE,并放到了独立进程,设置是否在stop的时候杀死这个进程,默认不杀死  
        option.SetIgnoreCacheException(false);//可选。默认false,设置是否收集CRASH信息,默认收集
option.setEnableSimulateGps(false);//可选,默认false,设置是否须要过滤gps仿真结果,默认须要
        mLocationClient.setLocOption(option);
    }

高精度定位模式:这样的定位模式下,会同一时候使用网络定位和GPS定位,优先返回最高精度的定位结果;

低功耗定位模式:这样的定位模式下。不会使用GPS,仅仅会使用网络定位(Wi-Fi和基站定位)。

仅用设备定位模式:这样的定位模式下。不须要连接网络,仅仅使用GPS进行定位,这样的模式下不支持室内环境的定位。

第三步。实现BDLocationListener接口

BDLocationListener接口有1个方法须要实现: 1.接收异步返回的定位结果,參数是BDLocation类型參数。

public class MyLocationListener implements BDLocationListener {
 
        @Override
        public void onReceiveLocation(BDLocation location) {
            //Receive Location
            StringBuffer sb = new StringBuffer(256);
            sb.append("time : ");
            sb.append(location.getTime());
            sb.append("\nerror code : ");
            sb.append(location.getLocType());
            sb.append("\nlatitude : ");
            sb.append(location.getLatitude());
            sb.append("\nlontitude : ");
            sb.append(location.getLongitude());
            sb.append("\nradius : ");
            sb.append(location.getRadius());
            if (location.getLocType() == BDLocation.TypeGpsLocation){// GPS定位结果
                sb.append("\nspeed : ");
                sb.append(location.getSpeed());// 单位:公里每小时
                sb.append("\nsatellite : ");
                sb.append(location.getSatelliteNumber());
                sb.append("\nheight : ");
                sb.append(location.getAltitude());// 单位:米
                sb.append("\ndirection : ");
                sb.append(location.getDirection());// 单位度
                sb.append("\naddr : ");
                sb.append(location.getAddrStr());
                sb.append("\ndescribe : ");
                sb.append("gps定位成功");
 
            } else if (location.getLocType() == BDLocation.TypeNetWorkLocation){// 网络定位结果
                sb.append("\naddr : ");
                sb.append(location.getAddrStr());
                //运营商信息
                sb.append("\noperationers : ");
                sb.append(location.getOperators());
                sb.append("\ndescribe : ");
                sb.append("网络定位成功");
            } else if (location.getLocType() == BDLocation.TypeOffLineLocation) {// 离线定位结果
                sb.append("\ndescribe : ");
                sb.append("离线定位成功,离线定位结果也是有效的");
            } else if (location.getLocType() == BDLocation.TypeServerError) {
                sb.append("\ndescribe : ");
                sb.append("服务端网络定位失败。能够反馈IMEI号和大体定位时间到loc-bugs@baidu.com,会有人追查原因");
            } else if (location.getLocType() == BDLocation.TypeNetWorkException) {
                sb.append("\ndescribe : ");
                sb.append("网络不同导致定位失败。请检查网络是否通畅");
            } else if (location.getLocType() == BDLocation.TypeCriteriaException) {
                sb.append("\ndescribe : ");
                sb.append("无法获取有效定位根据导致定位失败,通常是因为手机的原因。处于飞行模式下通常会造成这样的结果,能够试着重新启动手机");
            }
sb.append("\nlocationdescribe : ");
                sb.append(location.getLocationDescribe());// 位置语义化信息
                List<Poi> list = location.getPoiList();// POI数据
                if (list != null) {
                    sb.append("\npoilist size = : ");
                    sb.append(list.size());
                    for (Poi p : list) {
                        sb.append("\npoi= : ");
                        sb.append(p.getId() + " " + p.getName() + " " + p.getRank());
                    }
                }
            Log.i("BaiduLocationApiDem", sb.toString());
        }

BDLocation类,封装了定位SDK的定位结果,在BDLocationListener的onReceive方法中获取。通过该类用户能够获取error code,位置的坐标,精度半径等信息。

详细方法请參考类參考。

获取error code:

public int getLocType ( )

返回值:

61 : GPS定位结果,GPS定位成功。

62 : 无法获取有效定位根据,定位失败。请检查运营商网络或者wifi网络是否正常开启,尝试又一次请求定位。

63 : 网络异常,没有成功向server发起请求,请确认当前測试手机网络是否通畅,尝试又一次请求定位。

65 : 定位缓存的结果。

66 : 离线定位结果。通过requestOfflineLocaiton调用时相应的返回结果。

67 : 离线定位失败。

通过requestOfflineLocaiton调用时相应的返回结果。

68 : 网络连接失败时,查找本地离线定位时相应的返回结果。

161: 网络定位结果,网络定位定位成功。

162: 请求串密文解析失败,通常是因为clientSO文件载入失败造成。请严格參照开发指南或demo开发,放入相应SO文件。

167: 服务端定位失败。请您检查是否禁用获取位置信息权限,尝试又一次请求定位。

502: key參数错误,请依照说明文档又一次申请KEY。

505: key不存在或者非法,请依照说明文档又一次申请KEY。

601: key服务被开发人员自己禁用,请依照说明文档又一次申请KEY。

602: key mcode不匹配。您的ak配置过程中安全码设置有问题,请确保:sha1正确,“;”分号是英文状态;且包名是您当前执行应用的包名,请依照说明文档又一次申请KEY。

501~700:key验证失败,请依照说明文档又一次申请KEY。

假设不能定位。请记住这个返回值,并到百度LBS开放平台论坛Andriod定位SDK版块中进行交流http://bbs.lbsyun.baidu.com/forum.php?mod=forumdisplay&fid=10 。若返回值是162~167。请将错误码、imei和定位时间反馈至loc-bugs@baidu.com,以便我们跟进追查问题。

第四步,開始定位

mLocationClient.start();

start:启动定位SDK。 stop:关闭定位SDK。调用start之后仅仅须要等待定位结果自己主动回调就可以。

开发人员定位场景假设是单次定位的场景。在收到定位结果之后直接调用stop函数就可以。

假设stop之后仍然想进行定位,能够再次start等待定位结果回调就可以。

假设开发人员想依照自己逻辑请求定位,能够在start之后依照自己的逻辑请求locationclient.requestLocation()函数,会主动触发定位SDK内部定位逻辑。等待定位回调就可以。

位置提醒使用

位置提醒最多提醒3次。3次过后将不再提醒。 假如须要再次提醒,或者要改动提醒点坐标,都可通过函数SetNotifyLocation()来实现。

//位置提醒相关代码
mNotifyer = new NotifyLister();
mNotifyer.SetNotifyLocation(42.03249652949337,113.3129895882556,3000,"gps");//4个參数代表要位置提醒的点的坐标。详细含义依次为:纬度,经度,距离范围,坐标系类型(gcj02,gps,bd09,bd09ll)
mLocationClient.registerNotify(mNotifyer);
//注冊位置提醒监听事件后。能够通过SetNotifyLocation 来改动位置提醒设置。改动后立马生效。
//BDNotifyListner实现
public class NotifyLister extends BDNotifyListener{
       public void onNotify(BDLocation mlocation, float distance){
 	   mVibrator01.vibrate(1000);//振动提醒已到设定位置附近
       }
    }
//取消位置提醒
mLocationClient.removeNotifyEvent(mNotifyer);

四.个人理解

通过第三部分的说明,这三个类之间的关系也是很easy理解的。

LocationClient类作为SDK核心类。对定位服务进行一系列管理和操作,比方开启和停止定位;

LocationClientOption类是配置信息类。设置定位服务的配置信息。比方定位时间间隔、是否使用GPS等等。

BDLocationListener接口就可以获得定位数据。那么在该接口内能够对定位数据进行处理、或者存储等等;

      在实现定位个过程其中,可能会出现非常多问题。比方不显示地图、不能定位、不进入回调函数等等。网上的方法五花八门。可是真的能解决自己的问题吗?未必。检查API_KEY、检查LocationClient是否在主线程中声明等等,另外一个原因就是百度地图定位SDK更新非常快。15年还能找到的类。如今可能已经被覆盖了,所以实在不行,最好的方法就是移植百度的Demo



Android studio 百度地图开发(2)地图定位