由于某些原因Google的网络服务在中国不可访问,从而导致网络定位方式的API失效,而GPS定位虽然不用网络,但是必须在室外才能进行有效的定位。基于以上原因可以使用第三方公司的SDK,国内在这领域做得比较好的是百度、高德

一、申请API Key

要想使用百度的LBS功能,首先必须申请一个API Key。需要百度账号才能申请,没有的话可以去注册一个。登录你的百度账号,打开http://developer.baidu.com/user/reg链接,在里面填写一些注册信息即可(只需填写带 * 的部分内容就足够了),然后点击提交,进行邮箱验证,点击邮件发来的链接就可完成注册。

到此一切顺利!这样说明成为了一名百度开发者。接着访问http://lbsyun.baidu.com/apiconsole/key。目前新注册的用户列表是空的,点击创建应用就可以去申请APIKey了。应用名称可以随便填(最好和Android项目名称相同),应用类型选择Android SDK ,启用服务默认的就可以。

那么发布版和开发版SHA1是什么东西呢? 打开你的Android Stuidio 项目,点击Android stuidio右侧工具栏的Gradle →项目名称→:app→Tasks→android,这里展示了项目中所有内置的Gradle Tasks,其中signingReport这个Task可以用来查看签名文件信息。双击文件,结果如图所示.查看下方的SHA1即可。注意:我们使用的是debug.keystore文件生成的指纹。这是Android自动生成的一个用于测试的签名文件。而当应用程序发布时还需要创建一个正式的签名文件,如果需要得到它:可以在cmd中输入keytool -list -v -keystore <签名文件路径> 然后输入密码...

Android studio如何导入包含新版步骑行导航仅支持aar开发包格式的地图包_android

我们得到了SHA1指纹就是开发版的SHA1,但是我们暂时还没有发布版的,两个值都填一样就可以了。最后还有一个包名,写我们的应用程序的包名,比如com.example.hu.tourismsystem 。然后点击提交就创建成功了。然后有了 访问应用(AK)就可以完成后续LBS 开发工作了。


二、使用百度定位,引入包

没有Android项目的可以现在创建,不过项目包名要和API key的包名一致。准备LBS SDK : 下载地址是http://lbsyun.baidu.com/sdk/download,我们这次会用到基础地图和基础定位这两个SDK,将他们勾上,然后点击“开发包”下载。下载完后解压,文件内容如图

Android studio如何导入包含新版步骑行导航仅支持aar开发包格式的地图包_ide_02

将BaiduLBS_Android.jar 复制到 项目app模块下的libs 。右键src/main目录创建 Directory 取名为jniLibs的目录,把压缩包里的所有文件夹复制到这里,并且复制到app/libs下各一份。系统会在app/build.gradle默认有如下声明:

dependencies {
    compile fileTree(include: ['*.jar'], dir: 'libs') ...}

并且在buildTypes块下添加如下代码

sourceSets {
    main() {
        jniLibs.srcDirs = ['libs']
    }
}

找到AndroidStuidio顶部工具栏Sync按钮(sync Project with Gradle files 一般在从右数第四个),点击他。然后薄酒引用成功了,如图。这样我们就把LBS的SDK准备好了。

Android studio如何导入包含新版步骑行导航仅支持aar开发包格式的地图包_百度地图_03

注意:ksoap2-android... 这个与本项目无关

三、确定自己的经纬度:

首先在Manifest添加权限

<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标签下添加:

<meta-data android:name="com.baidu.lbsapi.API_KEY"
    android:value="KLqE1SiyhO90Osu7ruCO67cbJdidDrgn"/>
<service android:name="com.baidu.location.f" android:enabled="true"
    android:process=":remote" >

其中values 为申请到的Api Key,根据自己情况修改。


布局文件很简单:一个id为positon_textview的TextView。

主程序:

public class ActivityMicroTourism extends AppCompatActivity {
   
    public LocationClient mlocationClient;
    private TextView positionText;
    
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        
        setContentView(R.layout.activity_micro_tourism);
       
	mlocationClient = new LocationClient(getApplicationContext());
        mlocationClient.registerLocationListener(new MyLocationListener());
 	positionText = (TextView) findViewById(R.id.position_text_view);
        List<String> permissionList = new ArrayList<>();
        if (ContextCompat.checkSelfPermission(ActivityMicroTourism.this, Manifest.permission.ACCESS_FINE_LOCATION) != PackageManager.PERMISSION_GRANTED) {
            permissionList.add(Manifest.permission.ACCESS_FINE_LOCATION);
        }
        if (ContextCompat.checkSelfPermission(ActivityMicroTourism.this, Manifest.permission.READ_PHONE_STATE) != PackageManager.PERMISSION_GRANTED) {
            permissionList.add(Manifest.permission.READ_PHONE_STATE);
        }
        if(ContextCompat.checkSelfPermission(ActivityMicroTourism.this, Manifest.permission.WRITE_EXTERNAL_STORAGE)!= PackageManager.PERMISSION_GRANTED){
            permissionList.add(Manifest.permission.WRITE_EXTERNAL_STORAGE);
        }
        if (!permissionList.isEmpty()){
            String [] permissions =permissionList.toArray(new String[permissionList.size()]);
            ActivityCompat.requestPermissions(ActivityMicroTourism.this,permissions,1);
        }
        else{requestLocation();}
    }
    private void requestLocation(){
        mlocationClient.start();
    }
    @Override
    public void onRequestPermissionsResult(int requestCode,  String[] permissions, int[] grantResults) {
        switch (requestCode){
            case 1:
                if(grantResults.length>0){
                    for(int result :grantResults){
                        if(result!=PackageManager.PERMISSION_GRANTED){
                            Toast.makeText(this,"必须同意所有权限才能使用本程序",Toast.LENGTH_SHORT).show();
                            finish();
                            return;
                        }
                    }
                    requestLocation();
                }else {
                    Toast.makeText(this,"发生未知错误",Toast.LENGTH_SHORT).show();
                    finish();
                }
                break;
            default:
        }
    }
  
      public  class MyLocationListener implements BDLocationListener{
        @Override
        public void onReceiveLocation(BDLocation bdLocation) {

            StringBuilder currentPositon =new StringBuilder();
            
            currentPositon.append("纬度: ").append(bdLocation.getLatitude()).append("  ");
            currentPositon.append("经线: ").append(bdLocation.getLongitude()).append("  ");
          /*currentPositon.append("国家: ").append(bdLocation.getCountry()).append("  ");
            currentPositon.append("省: ").append(bdLocation.getProvince()).append("  ");
            currentPositon.append("市: ").append(bdLocation.getCity()).append("  ");
            currentPositon.append("区: ").append(bdLocation.getDistrict()).append("  ");
            currentPositon.append("街道: ").append(bdLocation.getStreet()).append("  ");*/
            currentPositon.append("定位方式: ");
            if(bdLocation.getLocType()==BDLocation.TypeGpsLocation){
                currentPositon.append("GPS");
            }else if(bdLocation.getLocType()==BDLocation.TypeNetWorkLocation){
                currentPositon.append("Wifi");
            }
	positionText.setText(currentPositon);
        }
}

在手机上运行该程序查看效果,同意所有权限,如果出现数据则表示成功。(如果只出现了经线和纬线,并且值非常离谱,也许只是你的网络太差)可以自己在搞个刷新按钮添加单击事件。有三个权限是需要在运行时用户同意的,为啥要添加那么多权限?因为不给添不给用。本段代码总的来说就是权限添加和权限验证和获取位置信息。

如果在用户在快速移动中,怎样实时更新位置?添加或修改如下代码:

private void initLocation(){
    LocationClientOption option =new LocationClientOption();
    option.setScanSpan(5000);   
    option.setIsNeedAddress(true);
    mlocationClient.setLocOption(option);
}
private void requestLocation(){
    initLocation();
    mlocationClient.start();
}
@Override
protected void onDestroy() {
    super.onDestroy();
    mlocationClient.stop();
}

每隔5秒钟会更新一下当前位置。更新太快容易消耗手机电量让手机发热.setIsNeedAddress 能让获取丰富的位置信息,现在可以把onReceiveLocation里的注释取消了。

4、使用百度地图

布局文件:

<LinearLayout
                android:layout_width="match_parent"
                android:layout_height="match_parent"
                android:background="@color/color_2"
                android:orientation="vertical" >


                <com.baidu.mapapi.map.MapView
                    android:id="@+id/bmapView"
                    android:layout_width="match_parent"
                    android:layout_height="match_parent"
                    android:clickable="true"></com.baidu.mapapi.map.MapView>
		 <TextView
                    android:id="@+id/position_text_view"
                    android:layout_width="wrap_content"
                    android:layout_height="wrap_content"
		    android:visibility="gone"
                    android:text="!"
                    />
</LinearLayout>



主程序:修改

public class ActivityMicroTourism extends AppCompatActivity  {
    ...
    private MapView mapView; // --
     @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        SDKInitializer.initialize(getApplication());//--
        setContentView(R.layout.activity_micro_tourism);
        mapView=(MapView)findViewById(R.id.bmapView);//--
        


        mlocationClient = new LocationClient(getApplicationContext());
        mlocationClient.registerLocationListener(new MyLocationListener());
	...
	}
 
    @Override                
    protected void onPause() {
        super.onPause();
        mapView.onPause();
    }

    @Override
    protected void onResume() {
        super.onResume();
        mapView.onResume();
        
    }

    @Override
    protected void onDestroy() {
        super.onDestroy();
        mlocationClient.stop();
       mapView.onDestroy();  //--
        
    }
   
}



用手机运行程序,看看百度地图是否能成功显示出来,模拟器是不支持百度地图SDK的。这里注意一点SDKInitializer.initialize()要放在加载页面之前。


如果能成功在手机上显示地图,那么接下来的任务就是将地图移动到我的位置,并且能让我的位置显示在地图上.

public class ActivityMicroTourism extends AppCompatActivity  {
    ...
     private BaiduMap baiduMap;          //--
     private boolean isFristLocate=true; //--
     @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        SDKInitializer.initialize(getApplication());
        setContentView(R.layout.activity_micro_tourism);
        mapView=(MapView)findViewById(R.id.bmapView);
        baiduMap=mapView.getMap();          //--
	baiduMap.setMyLocationEnabled(true);//--
        mlocationClient = new LocationClient(getApplicationContext());
        mlocationClient.registerLocationListener(new MyLocationListener());
	...
	}
 
      private void navigateTo(BDLocation location){
        if(isFristLocate ){
            LatLng ll=new LatLng(location.getLatitude(),location.getLongitude());
            MapStatusUpdate update= MapStatusUpdateFactory.newLatLng(ll);
            baiduMap.animateMapStatus(update);
            update=MapStatusUpdateFactory.zoomTo(16f);
            baiduMap.animateMapStatus(update);
            isFristLocate=false;
            
        }

        MyLocationData.Builder locationBuilder=new MyLocationData.Builder();
        locationBuilder.latitude(location.getLatitude());
        locationBuilder.longitude(location.getLongitude());
        MyLocationData locationData=locationBuilder.build();
        baiduMap.setMyLocationData(locationData);
    }
    @Override
    protected void onDestroy() {
        super.onDestroy();
        mlocationClient.stop();
        mapView.onDestroy();
       baiduMap.setMyLocationEnabled(false); //--
    }

 	public  class MyLocationListener implements BDLocationListener{
        @Override  //--
        public void onReceiveLocation(BDLocation bdLocation) {   
         if (bdLocation.getLocType()==BDLocation.TypeGpsLocation || bdLocation.getLocType()==BDLocation.TypeNetWorkLocation){
                navigateTo(bdLocation);
            }
		...

        }

        @Override
        public void onConnectHotSpotMessage(String s, int i) {}
    }
   
}



其中,navaigateTo()是用来用来更新位置的,第一次打开程序的时候会更新位置

手机运行程序,查看效果。需要注意的几点:

1、在布局文件百度地图控件会报错,可以无视,模拟器上运行不了可以在手机上运行

2、由于可能某些人的手机比较旧,刚开始打开程序更新位置会失败,可以尝试添加一个刷新按钮,执行navaigateTo()函数

3、更新位置是一个耗时比较大的操作,放在主线程中容易造成线程阻塞,最好在子线程中运行

4、百度地图定位坐标会发生点偏移,坐标校正可以参考http://blog.sina.com.cn/s/blog_80a9926b0101ktoa.html

5、更多的开发指南可以参考 http://lbsyun.baidu.com