简单实现安卓里百度地图持续定位
这几天自己研究了关于地手机上面开发安卓地图的问题,发现百度官方示例demo讲解百度持续定位方面还是讲解的有些不清楚,本人研究了几次之后将其弄得更详细以便于让各位方便学习,有不足之处请在评论区指出,官方示例的网址是:http://lbsyun.baidu.com/index.php?title=android-locsdk/guide/v5-0
上面的网址已经将安卓简单配置百度地图环境讲解的很详细了,再次不做赘述了,此外,可能会有人发现
packagecom.example.andoridloca; importjava.util.List; importjava.util.Timer; importjava.util.TimerTask; importandroid.app.Activity; importandroid.content.ContentValues; importandroid.content.Intent; importandroid.database.Cursor; importandroid.os.Bundle; importandroid.os.Handler; importandroid.os.Message; importandroid.text.method.ScrollingMovementMethod; importandroid.util.Log; importandroid.view.Menu; importandroid.view.MenuItem; importandroid.view.View; importandroid.view.View.OnClickListener; importandroid.widget.Button; importandroid.widget.TextView; importandroid.widget.Toast; importcom.baidu.location.BDLocation; importcom.baidu.location.BDLocationListener; importcom.baidu.location.LocationClient; importcom.baidu.location.LocationClientOption; importcom.baidu.location.BDNotifyListener;//假如用到位置提醒功能,需要import该类 importcom.baidu.location.LocationClientOption.LocationMode; importcom.baidu.location.Poi; importcom.baidu.mapapi.SDKInitializer; importcom.baidu.mapapi.map.MapView; publicclassMainActivityextendsActivityimplementsOnClickListener{ MapViewmMapView=null; publicstaticfinalStringTAG="mian"; StringBuffersb=newStringBuffer(256); publicStringBuilderbuilder=newStringBuilder(); privateButtonbt1; privateTextViewtv1; privateDBtoolsDBhelper; booleanisOpenLocation=false; publicLocationClientmLocationClient=null; publicBDLocationListenermyListener=newMyLocationListener(); @Override protectedvoidonCreate(BundlesavedInstanceState){ super.onCreate(savedInstanceState); SDKInitializer.initialize(getApplicationContext()); setContentView(R.layout.activity_main); DBhelper=newDBtools(this); tv1=(TextView)findViewById(R.id.textView1); tv1.setMovementMethod(newScrollingMovementMethod()); bt1=(Button)findViewById(R.id.button1); bt1.setOnClickListener(this); mMapView=(MapView)findViewById(R.id.bmapView); mLocationClient=newLocationClient(getApplicationContext());//声明LocationClient类 mLocationClient.registerLocationListener(myListener);//注册监听函数 initLocation(); } privatevoidinitLocation(){ LocationClientOptionoption=newLocationClientOption(); option.setLocationMode(LocationMode.Hight_Accuracy );//可选,默认高精度,设置定位模式,高精度,低功耗,仅设备 option.setCoorType("bd09ll");//可选,默认gcj02,设置返回的定位结果坐标系 intspan=0; 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); } @Override protectedvoidonDestroy(){ super.onDestroy(); //在activity执行onDestroy时执行mMapView.onDestroy(),实现地图生命周期管理 mMapView.onDestroy(); } @Override protectedvoidonResume(){ super.onResume(); //在activity执行onResume时执行mMapView.onResume(),实现地图生命周期管理 mMapView.onResume(); } @Override protectedvoidonPause(){ super.onPause(); //在activity执行onPause时执行mMapView.onPause(),实现地图生命周期管理 mMapView.onPause(); } publicclassMyLocationListenerimplementsBDLocationListener{ @Override publicvoidonReceiveLocation(BDLocationlocation){ //ReceiveLocation StringBuffersb=newStringBuffer(256); sb.append("time:"); sb.append(location.getTime()); sb.append("\nerrorcode:"); 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定位成功"); }elseif(location.getLocType()==BDLocation.TypeNetWorkLocation){//网络定位结果 sb.append("\naddr:"); sb.append(location.getAddrStr()); //运营商信息 sb.append("\noperationers:"); sb.append(location.getOperators()); sb.append("\ndescribe:"); sb.append("网络定位成功"); }elseif(location.getLocType()==BDLocation.TypeOffLineLocation){//离线定位结果 sb.append("\ndescribe:"); sb.append("离线定位成功,离线定位结果也是有效的"); }elseif(location.getLocType()==BDLocation.TypeServerError){ sb.append("\ndescribe:"); sb.append("服务端网络定位失败,可以反馈IMEI号和大体定位时间到loc-bugs@baidu.com,会有人追查原因"); }elseif(location.getLocType()==BDLocation.TypeNetWorkException){ sb.append("\ndescribe:"); sb.append("网络不同导致定位失败,请检查网络是否通畅"); }elseif(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("\npoilistsize=:"); sb.append(list.size()); for(Poip:list){ sb.append("\npoi=:"); sb.append(p.getId()+""+p.getName()+""+p.getRank()); } } Log.i("BaiduLocationApiDem",sb.toString()); DBtoolsdbhelper=newDBtools(getApplicationContext()); ContentValuesinitialValues=newContentValues(); initialValues.put("shijian",location.getTime()); initialValues.put("didian",location.getLatitude()+"--"+location.getLongitude()); dbhelper.open(); dbhelper.insert("path",initialValues); dbhelper.close(); tv1.setText(sb.toString()); } } @Override publicvoidonClick(Viewarg0){ Threadmytime=newThread(newThreadShow()); if(isOpenLocation){ mLocationClient.stop(); isOpenLocation=false; } else{ Toast.makeText(getApplicationContext(),"开启",Toast.LENGTH_SHORT).show(); isOpenLocation=true; //mLocationClient.start(); mytime.start(); } } //handler类接收数据 Handlerhandler=newHandler(){ publicvoidhandleMessage(Messagemsg){ if(msg.what==1){ Log.i("BaiduLocationApiDem","加以"); mLocationClient.start(); mLocationClient.requestLocation(); } }; }; //线程类 classThreadShowimplementsRunnable{ @Override publicvoidrun(){ //TODOAuto-generatedmethodstub while(isOpenLocation){ try{ mLocationClient.stop(); Thread.sleep(2000); Messagemsg=newMessage(); msg.what=1; handler.sendMessage(msg); //System.out.println("send..."); }catch(Exceptione){ //TODOAuto-generatedcatchblock e.printStackTrace(); System.out.println("threaderror..."); } } } } }
这里面关于mLocationClient.stop();mLocationClient.start(); mLocationClient.requestLocation();这三个函数我有必要讲解一下,因为持续定位时这三个函数的配合使用很重要,官方文档里面解释说mLocationClient.start()函数用于开启定位,mLocationClient.requestLocation()函数用于主动触发定位SDK内部定位逻辑,个人感觉差不多,两个都会执行我的mLocationClient的所属类里面的逻辑代码,可能是我的项目就这样吧,然后是mLocationClient.stop(),此函数用于停止定位,如果持续定位的话,是需要和mLocationClient.start()函数配合使用的,具体在上面的代码里面有展示。
切记不要将mLocationClient.start()和mLocationClient.stop()一起使用,我在网上查询时好像是说一部原因,具体举一个例子吧:
//某段代码如果是这样的话按照逻辑韩式会将mLocationClient所属类的里面逻辑代码执行一遍,具体见MainAvtivity里面的MyLocationListener类内容,但是实际上是不会执行的 mLocationClient.start(); mLocationClient.stop();
所以在我的MainActivity里面我使用线程来一遍遍的执行start和stop函数,这样就会消除刚刚说的这种效果,最后就能够实现持续定位了。
在此给出我的布局文件配合看看
<LinearLayoutxmlns: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:orientation="vertical" tools:context="com.example.newloca.MainActivity"> <RelativeLayout android:layout_width="match_parent" android:layout_height="300dp"> <com.baidu.mapapi.map.MapView android:id="@+id/bmapView" android:layout_width="fill_parent" android:layout_height="fill_parent" android:clickable="true"/> </RelativeLayout> <TextView android:id="@+id/textView1" android:layout_width="match_parent" android:layout_height="150dp" android:scrollbars="vertical" android:background="#f00" android:text="位置信息"/> <RelativeLayout android:layout_width="match_parent" android:layout_height="match_parent" > <Button android:id="@+id/button1" android:layout_width="wrap_content" android:layout_height="wrap_content" android:layout_alignParentRight="true" android:layout_alignParentBottom="true" android:text="获取位置"/> </RelativeLayout> </LinearLayout>
其他像权限什么的配置,用最开始给的官方地址里面的就行了
顺便说一下,本人是使用的安卓4.2版本开发的,手机真机调试和虚拟机调试在定位的时间间隔上面会有点误差,也不知道什么原因
以上就是本文的全部内容,希望本文的内容对大家的学习或者工作能带来一定的帮助,同时也希望多多支持毛票票!