百度地图集成步骤及使用中遇到的问题

百度地图集成步骤:

一. 获取开发密钥(AK)

百度地图 SDK开发密钥的申请地址为:http://lbsyun.baidu.com/apiconsole/key

二. 在项目中集成BaiduMap SDK

1. 下载开发包

   选择“基础地图”,

2. 将开发包拷贝至工程

(1)添加jar文件,BaiduLBS_Android.jar文件将其拷贝至工程的app/libs目录下

(2)添加so文件

      有两种方法可以往项目中添加so文件。
     方法一:
    在下载的开发包中拷贝需要的CPU架构对应的so文件文件夹到app/libs目录下

在app目录下的build.gradle文件中android块中配置sourceSets标签,如果没有使用该标签则新增,详细配置代码如下:

sourceSets {
           main {
               jniLibs.srcDir 'libs'
           }
    }

方法二:
     在src/main/目录下新建jniLibs目录(如果您的项目中已经包含该目录不用重复创建),在下载的开发包中拷贝项目中需要的CPU架构对应的so文件文件夹到jniLibs目录

3. 编写混淆文件,打开app目录下的proguard-rules.pro文件,添加如下代码

-keep class com.baidu.** {*;}
-keep class mapsdkvi.com.** {*;}    
-dontwarn com.baidu.**

注意:保证百度类不能被混淆,否则会出现网络不可用等运行时异常

三. 显示地图并定位:

1. 配置AndroidManifest.xml

(1)在<application>中加入如下代码配置开发密钥(AK)

<application>  
    <meta-data  
        android:name="com.baidu.lbsapi.API_KEY"  
        android:value="开发者 key" />  
    <!-- 声明定位的service组件 -->
    <service android:name="com.baidu.location.f"
        android:enabled="true"
        android:process=":remote"/>
</application>

(2) 在<application/>外部添加如下权限声明:

<!-- 访问网络,进行地图相关业务数据请求,包括地图数据,路线规划,POI检索等 -->
<uses-permission android:name="android.permission.INTERNET" />
<!-- 获取网络状态,根据网络状态切换进行数据请求网络转换 -->
<uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" />
    
<!-- 读取外置存储。如果开发者使用了so动态加载功能并且把so文件放在了外置存储区域,则需要申请该权限,否则不需要 -->
<uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE" />
<!-- 写外置存储。如果开发者使用了离线地图,并且数据写在外置存储区域,则需要申请该权限 -->
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" /> 

<!-- 这个权限用于进行网络定位 -->
<uses-permission android:name="android.permission.ACCESS_COARSE_LOCATION" />
<!-- 这个权限用于访问GPS定位 -->
<uses-permission android:name="android.permission.ACCESS_FINE_LOCATION" />

2. 在布局文件中添加地图容器

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

3. 地图初始化

在项目的Application类的onCreate方法中完成SDK的初始化

//在使用SDK各组件之前初始化context信息,传入ApplicationContext   
        SDKInitializer.initialize(this);
        //自4.3.0起,百度地图SDK所有接口均支持百度坐标和国测局坐标,用此方法设置您使用的坐标类型.
        //包括BD09LL和GCJ02两种坐标,默认是BD09LL坐标。
        SDKInitializer.setCoordType(CoordType.BD09LL);

4. 在使用地图的Activity中,管理MapView生命周期

    private MapView bdMapView = null;  
    @Override  
    protected void onCreate(Bundle savedInstanceState) {  
        super.onCreate(savedInstanceState);    
        setContentView(R.layout.activity_main);  
        //获取地图控件引用  
        bdMapView = (MapView) findViewById(R.id.bmapView);  
        requestPermission();
        initMapData();
    }  
    @Override  
    protected void onResume() {  
       super.onResume();  
       //在activity执行onResume时执行mMapView. onResume (),实现地图生命周期管理  
       bdMapView.onResume();  
    }  
    @Override  
    protected void onPause() {  
      super.onPause();  
      //在activity执行onPause时执行mMapView. onPause (),实现地图生命周期管理  
      bdMapView.onPause();  
    } 
    @Override  
    protected void onDestroy() {  
      super.onDestroy();  
      //在activity执行onDestroy时执行mMapView.onDestroy(),实现地图生命周期管理  
      if (bdMapView != null) {
            // 退出时销毁定位
            locationClient.unRegisterLocationListener(myLocationListener);
            locationClient.stop();
            // 关闭定位图层
            mBaiduMap.setMyLocationEnabled(false);
            bdMapView.onDestroy();
            bdMapView = null;
        }  
    }  

5. 进入页面,首先申请动态权限

private void requestPermission() {
        if (Build.VERSION.SDK_INT >= 23 ) {

//            isPermissionRequested = true;
            ArrayList<String> permissionsList = new ArrayList<>();
            String[] permissions = {
                    Manifest.permission.ACCESS_NETWORK_STATE,
                    Manifest.permission.INTERNET,
                    Manifest.permission.ACCESS_COARSE_LOCATION,
                    Manifest.permission.ACCESS_FINE_LOCATION,
                    Manifest.permission.CAMERA,
                    Manifest.permission.READ_EXTERNAL_STORAGE,
                    Manifest.permission.WRITE_EXTERNAL_STORAGE,
            };

            for (String perm : permissions) {
                if (PackageManager.PERMISSION_GRANTED != checkSelfPermission(perm)) {
                    permissionsList.add(perm);
                    // 进入到这里代表没有权限.
                }
            }

            if (!permissionsList.isEmpty()) {
                requestPermissions(permissionsList.toArray(new String[permissionsList.size()]), 0);
            }
        }
    }

6. 初始化地图

private BaiduMap mBaiduMap;
private LocationClient locationClient;
private MapStatusUpdate update;
private MyLocationListener myLocationListener;
private void initMapData() {
        mBaiduMap = bdMapView.getMap();
        mBaiduMap.setMapType(BaiduMap.MAP_TYPE_NORMAL);
        //开启地图的定位图层
        mBaiduMap.setMyLocationEnabled(true);
        
        //地图打印信息中定位成功,但一直显示在北京
        //要先设置缩放比例,再进行经纬度的显示,最终就可以定位自己
        // 在MyLocationListener中设置经纬度
        update = MapStatusUpdateFactory.zoomTo(16f);
        mBaiduMap.animateMapStatus(update);


        //通过LocationClient发起定位
        locationClient = new LocationClient(getApplicationContext());
        LocationClientOption option = new LocationClientOption();
        option.setOpenGps(true);
        //可选,设置是否需要地址信息,默认不需要
        option.setIsNeedAddress(true);
        option.setPriority(LocationClientOption.NetWorkFirst); // 设置网络优先
        //可选,默认高精度,设置定位模式,高精度,低功耗,仅设备
        option.setLocationMode(LocationClientOption.LocationMode.Hight_Accuracy);
        //可选,默认gcj02,设置返回的定位结果坐标系
        option.setCoorType("bd09ll");
        //可选,默认0,即仅定位一次,设置发起定位请求的间隔需要大于等于1000ms才是有效的
        option.setScanSpan(1000);
        locationClient.setLocOption(option);

        myLocationListener = new MyLocationListener();
        locationClient.registerLocationListener(myLocationListener);

        //开启地图定位
        locationClient.start();

        mBaiduMap.setMyLocationConfiguration(new MyLocationConfiguration(
                MyLocationConfiguration.LocationMode.NORMAL, true, BitmapDescriptorFactory.fromResource(R.mipmap.icon_gcoding),
                0x0f1679b3, 0xAA00FF00));
    }



    //通过继承抽象类BDAbstractListener并重写其onReceieveLocation方法来获取定位数据,并将其传给MapView
    public class MyLocationListener extends BDAbstractLocationListener {

        @Override
        public void onReceiveLocation(BDLocation bdLocation) {
            if (bdLocation == null || bdMapView == null) {
                return;
            }

            String province = bdLocation.getProvince();    //获取省份
            String city = bdLocation.getCity();    //获取城市
            String district = bdLocation.getDistrict();    //获取区县
            String street = bdLocation.getStreet();    //获取街道信息
            String mAddress = province + city + district + street;
            //获取纬度信息
            double mLatitude = bdLocation.getLatitude();
            //获取经度信息
            double mLongitude = bdLocation.getLongitude();
            float radius = bdLocation.getRadius();    //获取定位精度,默认值为0.0f

            String coorType = bdLocation.getCoorType();
            Log.d(TAG, "bdLocation="+bdLocation+",mAddress="+mAddress+","+radius+","+coorType);
            MyLocationData locationData = new MyLocationData.Builder()
                    .accuracy(bdLocation.getRadius())
                    .direction(bdLocation.getDirection())
//                    .direction(100)
                    .latitude(bdLocation.getLatitude())
                    .longitude(bdLocation.getLongitude())
                    .build();

            mBaiduMap.setMyLocationData(locationData);
            //地图打印信息中定位成功,但一直显示在北京
            //要先设置缩放比例,再进行经纬度的显示,最终就可以定位自己
            //在init中设置了缩放比例,在这里进行经纬度的显示
            LatLng latLng = new LatLng(mLatitude, mLongitude);
            update = MapStatusUpdateFactory.newLatLng(latLng);
            mBaiduMap.animateMapStatus(update);
        }
    }

四.  使用中遇到的问题:

1. 打包后,第一次定位成功,再次定位没有定位成功,报locType:505

      原因:505错误码的意思是AK错误

      解决方法:使用keytool -list -v -keystore G:\project\nmpaapp\nmpa.jks -alias adcd或使用signingReport方法获取正确的SHA1,配置上即可。

      思路:  获取发布版SHA1时,先是使用命令keytool -list -v -keystore G:\project\nmpaapp\nmpa.jks -alias adcd 获取了SHA1,但看到开发者文档里没有设置别名,就把别名去掉了,重新获取了一次,结果配置上去后,第一次能定位成功,闪一下正确的定位后,就显示空页面了。

在网上查看了获取SHA1码的资料,在build.gradle文件中配置了signingConfigs后,

signingConfigs {
        //你自己的keystore信息
        myConfig {
            storeFile file("../nmpa.jks")
            storePassword "123456"
            keyAlias "adcd"
            keyPassword "123456"
            v2SigningEnabled false
        }
}


buildTypes {
        release {
            //是否混淆
            minifyEnabled false
            //Zipalign优化
            zipAlignEnabled true
            // 移除无用的resource文件
            shrinkResources false
            //签名文件
            signingConfig signingConfigs.myConfig
            proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro'
        }
    }

打开android studio的右侧边栏"Gradle",找到app--->android---->signingReport,双击运行,会出现debug版本和release版本的SHA1,

> Task :app:signingReport
Variant: debug
Config: debug
Store: C:\Users\Administrator\.android\debug.keystore
Alias: AndroidDebugKey
MD5: 
SHA1: 
SHA-256: 
Valid until: 2050年3月4日 星期五

Variant: release
Config: myConfig
Store: G:\project\nmpaapp\nmpa.jks
Alias: adcd
MD5: 
SHA1: 
SHA-256: 
Valid until: 2045年4月11日 星期二

最终发现:使用signingReport方式获取到的发布版SHA1与一开始使用命令keytool -list -v -keystore G:\project\nmpaapp\nmpa.jks -alias adcd获取到的SHA1是一样的。

2. 刚开始运行时,定位不成功,原因:没有动态申请位置权限;

3. 打印信息中显示定位成功,但地图上还是显示北京的地图:

   原因:要先设置缩放比例,再进行经纬度的显示,最终就可以定位自己

   解决:在class MyLocationListener extends BDAbstractLocationListener监听中添加

        MapStatusUpdate update = MapStatusUpdateFactory.zoomTo(16f);
        mBaiduMap.animateMapStatus(update);
        LatLng latLng = new LatLng(mLatitude, mLongitude);
        update = MapStatusUpdateFactory.newLatLng(latLng);
        mBaiduMap.animateMapStatus(update);

 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值