由于某些原因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 <签名文件路径> 然后输入密码...
我们得到了SHA1指纹就是开发版的SHA1,但是我们暂时还没有发布版的,两个值都填一样就可以了。最后还有一个包名,写我们的应用程序的包名,比如com.example.hu.tourismsystem 。然后点击提交就创建成功了。然后有了 访问应用(AK)就可以完成后续LBS 开发工作了。
二、使用百度定位,引入包
没有Android项目的可以现在创建,不过项目包名要和API key的包名一致。准备LBS SDK : 下载地址是http://lbsyun.baidu.com/sdk/download,我们这次会用到基础地图和基础定位这两个SDK,将他们勾上,然后点击“开发包”下载。下载完后解压,文件内容如图
将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准备好了。
注意: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