准备工作:

1.申请百度地图API

2.下载百度地图的SDK

3.将SDK包中的BaiduLBS_Android.jar文件放到,项目里的app/libs里面

4.在src/main目录下创建一个名为jniLibs的目录,然后将SDK包中的其他所有目录放在里面。

5.同步一下项目。

 

首先修改布局文件

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:orientation="vertical"
    android:layout_width="match_parent"
    android:layout_height="match_parent">

    <TextView
        android:id="@+id/position_text_view"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content" />
</LinearLayout>

用一个TextView控件用于显示经纬度。

 

然后修改AndroidManifest.xml文件,代码如下:

<manifest xmlns:android="http://schemas.android.com/apk/res/android"
    package="co.example.leo.lbstest">

    <uses-permission android:name="android.permission.ACCESS_COARSE_LOCATION"/>
    <uses-permission android:name="android.permission.ACCESS_FINE_LOCATION"/>
    <uses-permission android:name="android.permission.ACCESS_WIFI_STATE"/>
    <uses-permission android:name="android.permission.ACCESS_NETWORK_STATE"/>
    <uses-permission android:name="android.permission.CHANGE_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.INTERNET"/>
    <uses-permission android:name="android.permission.MOUNT_UNMOUNT_FILESYSTEMS"/>
    <uses-permission android:name="android.permission.WAKE_LOCK"/>

    <application
        android:allowBackup="true"
        android:icon="@mipmap/ic_launcher"
        android:label="@string/app_name"
        android:roundIcon="@mipmap/ic_launcher_round"
        android:supportsRtl="true"
        android:theme="@style/AppTheme">
        <meta-data
            android:name="com.baidu.lbsapi.API_KEY"
            android:value="N5yXI1cjwQXw631GaapHpD5Fdv43t8KW"/>
        <activity android:name=".MainActivity">
            <intent-filter>
                <action android:name="android.intent.action.MAIN" />

                <category android:name="android.intent.category.LAUNCHER" />
            </intent-filter>
        </activity>

        <service android:name="com.baidu.location.f"
            android:exported="true"
            android:process=":remote">
        </service>

    </application>

</manifest>

里面添加了很多的权限,每一个都是百度LBS SDK内部要用到的, 然后在<application>标签内部添加了一个<meta -data>标签。

这个标签的androi:name部分是固定的,必须填写com.baidu.lbsapi.API_KEY

android:value部分填入申请的APIKey。

最后还要对LBS SDK中的服务进行注册。

 

 

最后修改MainActivity中的代码。

  1 import android.Manifest;
  2 import android.content.pm.PackageManager;
  3 import android.os.Bundle;
  4 import android.support.annotation.NonNull;
  5 import android.support.v4.app.ActivityCompat;
  6 import android.support.v4.content.ContextCompat;
  7 import android.support.v7.app.AppCompatActivity;
  8 import android.widget.TextView;
  9 import android.widget.Toast;
 10 
 11 import com.baidu.location.BDLocation;
 12 import com.baidu.location.BDLocationListener;
 13 import com.baidu.location.LocationClient;
 14 
 15 import java.util.ArrayList;
 16 import java.util.List;
 17 
 18 public class MainActivity extends AppCompatActivity {
 19 
 20     public LocationClient mLocationClient;
 21     private TextView positionText;
 22 
 23     @Override
 24     protected void onCreate(Bundle savedInstanceState) {
 25         super.onCreate(savedInstanceState);
 26         //创建一个LocationClient的实例,接受的context通过getApplicationContext()方法获取。
 27         mLocationClient = new LocationClient(getApplicationContext());
 28         //调用LocationClient的registerLocationListener()方法来注册一个监听器 当获取到位置信息的时候,就会回调这个定位监听器
 29         mLocationClient.registerLocationListener(new MyLocationListener());
 30         setContentView(R.layout.activity_main);
 31 
 32         positionText = (TextView)findViewById(R.id.position_text_view);
 33         /*
 34         * 之前在AndroidManifest.xml内声明了很多权限。
 35         * 其中有4个是危险权限。不过ACCESS_COARSE_LOCATION 和 ACCESS_FINE_LOCATION都属于一个权限组,所以两者只需要申请其中一个就可以了。
 36         * 如何在运行时一次申请三个权限呢?
 37         * 首先创建一个空的List集合,然后依次判断这三个权限有没有被授权,如果没有授权就添加到List集合中,最后将List集合转化成数组,在调用ActivityCompat.requestPermissions()方法就可以一次性申请。
 38          */
 39         List<String> permissionList = new ArrayList<>();
 40         if(ContextCompat.checkSelfPermission(this, Manifest.permission.ACCESS_FINE_LOCATION) != PackageManager.PERMISSION_GRANTED){
 41             permissionList.add(Manifest.permission.ACCESS_FINE_LOCATION);
 42         }
 43         if(ContextCompat.checkSelfPermission(this,Manifest.permission.READ_PHONE_STATE) != PackageManager.PERMISSION_GRANTED){
 44             permissionList.add(Manifest.permission.READ_PHONE_STATE);
 45         }
 46         if(ContextCompat.checkSelfPermission(this,Manifest.permission.WRITE_EXTERNAL_STORAGE) != PackageManager.PERMISSION_GRANTED){
 47             permissionList.add(Manifest.permission.WRITE_EXTERNAL_STORAGE);
 48         }
 49         if(! permissionList.isEmpty()){
 50             String[] permissions = permissionList.toArray(new String[permissionList.size()]);
 51             ActivityCompat.requestPermissions(this,permissions,1);
 52         }else{
 53             requestLocation();
 54         }
 55     }
 56 
 57     private void requestLocation(){
 58         //调用start方法会回调到我们注册的监听器上面
 59         mLocationClient.start();
 60     }
 61 
 62     @Override
 63     public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) {
 64         switch (requestCode){
 65             case 1:
 66                 if(grantResults.length > 0){
 67                     //将每个申请的权限都进行判断 如果存在一个没有被授权,那么就调用finish()方法关闭程序。
 68                     for(int result : grantResults){
 69                         if(result != PackageManager.PERMISSION_GRANTED){
 70                             Toast.makeText(MainActivity.this,"必须同意所有权限才能使用本程序",Toast.LENGTH_SHORT).show();
 71                             finish();
 72                             return ;
 73                         }
 74                     }
 75                     //所有权限都已经授权,那么直接调用requestLocation()方法开始定位。
 76                     requestLocation();
 77                 }else{
 78                     Toast.makeText(MainActivity.this,"发生未知错误", Toast.LENGTH_SHORT).show();
 79                     finish();
 80                 }
 81                 break;
 82             default: break;
 83         }
 84     }
 85 
 86     public class MyLocationListener implements BDLocationListener{
 87         @Override
 88         public void onReceiveLocation(final BDLocation location) {
 89             runOnUiThread(new Runnable() {
 90                 @Override
 91                 public void run() {
 92                     StringBuilder currentPosition = new StringBuilder();
 93                     //通过BDLocation的getLatitude()方法获取当前位置的纬度
 94                     currentPosition.append("纬度").append(location.getLatitude()).append("\n");
 95                     //通过BDLocation的getLongitude()方法获取当前位置的经度。
 96                     currentPosition.append("经线").append(location.getLongitude()).append("\n");
 97                     //getLocType()方法获取当前的定位方式。
 98                     if(location.getLocType() == BDLocation.TypeGpsLocation){
 99                         currentPosition.append("GPS");
100                     }else if(location.getLocType() == BDLocation.TypeNetWorkLocation){
101                         currentPosition.append("网络");
102                     }
103                     positionText.setText(currentPosition);
104                 }
105             });
106         }
107 
108         @Override
109         public void onConnectHotSpotMessage(String s, int i) {
110 
111         }
112     }
113 }

 

默认情况下,调用Location的start()方法只会定位一次。

如果想要实时更新当前的位置怎么办呢?

修改MainActivity中的代码

1 private void requestLocation(){
2         //调用start方法会回调到我们注册的监听器上面
3         initLocation();
4         mLocationClient.start();
5     }

这里在requestLocation()方法内增加了一个initLocation()方法

1  private void initLocation(){
2         LocationClientOption option = new LocationClientOption();
3         option.setScanSpan(5000);
4         mLocationClient.setLocOption(option);
5     }

在initLocation()方法中创建了一个LocationClientOption对象,然后调用它的setScanSpan()方法来设置更新的间隔。单位为毫秒。

最后记得,在活动销毁的时候要调用LocationClient的Stop()方法来停止定位。

1  @Override
2         protected void onDestroy(){
3             super.onDestroy();
4             mLocationClient.stop();
5         }

 

选择定位模式

定位模式有三种:

高精确度:允许使用GPS,无线网络,蓝牙或者移动网络来进行定位。

节电:仅允许使用WLAN,蓝牙或移动网络确定位置。

仅限设备:仅限GPS来进行定位。

也就是说,如果要想使用GPS定位功能,那么必须要选择高精确度模式,或者是仅限设备模式。

我们可以在之前的initLocation()方法中对百度LBS SDK的定位模式进行指定。

一共有三种模式可选:High_Accuracy,Battery_Saving,和Device_Sensors。

Hight_Accuracy表示高精确度模式,会在GPS信号正常的情况下优先使用GPS定位。

Device_Sensors表示传感器模式,只会用GPS进行定位。

High_Accuracy是默认的模式。

 

修改initLocation()中的代码

1 private void initLocation(){
2         LocationClientOption option = new LocationClientOption();
3         option.setScanSpan(5000);
4         option.setLocationMode(LocationClientOption.LocationMode.Device_Sensors);
5         mLocationClient.setLocOption(option);
6     }

这里调用了setLocationMode()方法将定位模式指定成传感器模式,也就是说只能用GPS进行定位。

 

看的懂的位置信息

我们可以通过一些简单的接口调用就能得到当前位置各种丰富的地址信息。

修改MainActivity中的initLocation()方法

 

1  private void initLocation(){
2         LocationClientOption option = new LocationClientOption();
3         option.setScanSpan(5000);
4         option.setIsNeedAddress(true);
5         mLocationClient.setLocOption(option);
6     }

 

 

 

 

这里调用了LocationClientOption的setIsNeedAddress()方法,并传入了true。这就表示我们需要获取当前位置的详细的地址信息。

然后修改MainActivity中的MyLocationListener

 1 public class MyLocationListener implements BDLocationListener{
 2         @Override
 3         public void onReceiveLocation(final BDLocation location) {
 4             runOnUiThread(new Runnable() {
 5                 @Override
 6                 public void run() {
 7                     StringBuilder currentPosition = new StringBuilder();
 8                     //通过BDLocation的getLatitude()方法获取当前位置的纬度
 9                     currentPosition.append("纬度").append(location.getLatitude()).append("\n");
10                     //通过BDLocation的getLongitude()方法获取当前位置的经度。
11                     currentPosition.append("经线").append(location.getLongitude()).append("\n");
12                     currentPosition.append("国家").append(location.getCountry()).append("\n");
13                     currentPosition.append("省").append(location.getProvince()).append("\n");
14                     currentPosition.append("市").append(location.getCity()).append("\n");
15                     currentPosition.append("区").append(location.getDistrict()).append("\n");
16                     currentPosition.append("街道").append(location.getStreet()).append("\n");
17                     //getLocType()方法获取当前的定位方式。
18                     if(location.getLocType() == BDLocation.TypeGpsLocation){
19                         currentPosition.append("GPS");
20                     }else if(location.getLocType() == BDLocation.TypeNetWorkLocation){
21                         currentPosition.append("网络");
22                     }
23                     positionText.setText(currentPosition);
24                 }
25             });
26         }
getCounty()获取当前所在国家
getProvince()获取当前所在省

以此类推。

需要我们注意的是获取地址信息一定需要用到网络,因此即使我们将定位模式指定成了Device Sensors,也会自动开启网络定位功能。

 

让“我”显示在地图上

百度LBS SDK当中提供了一个MyLocatioData.Builder类,这个类是用来封装设备当前所在位置的。只需要将经纬度信息传入到这个类的相应方法中就可以了。如下所示:

MyLocationData.Builder locationBuilder = new MyLocationData.Builder();
locationBuilder.latitude(39.315);
locationBuilder.longitude(116.404);

MyLocationData.Builder类中还提供了一个builde()方法,当我们把要封装的信息都设置完成之后,只需要调用它的build()方法。就会生成一个MyLocationData的实例。

然后将这个实例传入到BaiduMap的setMyLocationData()方法中,就可以让设备当前的位置显示在地图上了。