需求:
每间隔10秒,同时距离移动大于1米的情况下,记录一次位置信息,在地图上绘制60秒的行动轨迹。
前提:
环境:Android Studio 、Windows10、Android手机
高德API的引用可见我上一篇博客:https://www.jianshu.com/p/4b773727e42c
在上一篇博客已经实现了地图的显示,因此需要实现的功能如下:
1.记录位置
2.在地图上绘制轨迹
一、添加Service组件
在AndroidManifest.xml的application标签中声明service组件,每个app拥有自己单独的定位service。声明如下:
<service android:name="com.amap.api.location.APSService"></service>
二、修改MainActivity.java
主要是定位+回调函数绘制路线。
public class MainActivity extends Activity implements LocationSource,AMapLocationListener {
MapView mMapView = null;
AMap aMap = null;
private AMapLocation oldLocation;
List<LatLng> points = new ArrayList<LatLng>();
Polyline polyline;
public int count = 0;
private OnLocationChangedListener mListener;
private AMapLocationClient mlocationClient;
private AMapLocationClientOption mLocationOption;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
mMapView = (MapView) findViewById(R.id.map);
mMapView.onCreate(savedInstanceState);// 此方法必须重写
init();
}
private void init(){
if (aMap == null) {
aMap = mMapView.getMap();
}
//设置amap的一些属性
UiSettings uiSettings = aMap.getUiSettings();
uiSettings.setCompassEnabled(true);// 设置指南针是否显示
uiSettings.setZoomControlsEnabled(true);// 设置缩放按钮是否显示
uiSettings.setScaleControlsEnabled(true);// 设置比例尺是否显示
uiSettings.setRotateGesturesEnabled(true);// 设置地图旋转是否可用
uiSettings.setTiltGesturesEnabled(true);// 设置地图倾斜是否可用
uiSettings.setMyLocationButtonEnabled(false);// 设置默认定位按钮是否显示
MyLocationStyle myLocationStyle = new MyLocationStyle();
myLocationStyle.myLocationType(MyLocationStyle.LOCATION_TYPE_LOCATION_ROTATE_NO_CENTER);//连续定位、蓝点不会移动到地图中心点,定位点依照设备方向旋转,并且蓝点会跟随设备移动。
aMap.setMyLocationStyle(myLocationStyle);
aMap.setLocationSource(this);// 设置定位监听
aMap.setMyLocationEnabled(true);// 设置为true表示显示定位层并可触发定位,false表示隐藏定位层并不可触发定位,默认是false
// 设置定位的类型为定位模式 ,可以由定位 LOCATION_TYPE_LOCATE、跟随 LOCATION_TYPE_MAP_FOLLOW 或地图根据面向方向旋转 LOCATION_TYPE_MAP_ROTATE
aMap.setMyLocationType(AMap.LOCATION_TYPE_MAP_ROTATE);
// 缩放级别(zoom):值越大地图越详细(4-20)
aMap.moveCamera(CameraUpdateFactory.zoomTo(16));
//使用 aMap.setMapTextZIndex(2) 可以将地图底图文字设置在添加的覆盖物之上
aMap.setMapTextZIndex(2);
}
/**
* 定位成功后回调函数
*/
@Override
public void onLocationChanged(AMapLocation amapLocation) {
if (mListener != null && amapLocation != null) {
if (amapLocation.getErrorCode() == 0) {
mListener.onLocationChanged(amapLocation);// 显示系统小蓝点
Log.e("Amap", amapLocation.getLatitude() + "," + amapLocation.getLongitude());
LatLng newLatLng = new LatLng(amapLocation.getLatitude(),amapLocation.getLongitude());
points.add(count, newLatLng);
count++;
drawLines();
} else {
String errText = "定位失败," + amapLocation.getErrorCode()+ ": " + amapLocation.getErrorInfo();
Log.e("AmapErr", errText);
Toast.makeText(this, errText, Toast.LENGTH_SHORT).show();
}
}
}
/**
* 绘制路线
*/
public void drawLines() {
PolylineOptions options = new PolylineOptions();
options.setCustomTexture(BitmapDescriptorFactory.fromResource(R.drawable.running));
options.geodesic(true).setDottedLine(false).color(Color.GREEN).addAll(points).useGradient(true).width(10).visible(true);
polyline = aMap.addPolyline(options);
// 获取轨迹坐标点
LatLngBounds.Builder b= LatLngBounds.builder();
for (int i = 0; i< points.size(); i++) {
b.include(points.get(i));
}
LatLngBounds bounds= b.build();
CameraUpdate update = CameraUpdateFactory.newLatLngBounds(bounds, 100);//
aMap.animateCamera(update);
}
/**
* 激活定位
*/
@Override
public void activate(OnLocationChangedListener listener) {
mListener = listener;
if (mlocationClient == null) {
mlocationClient = new AMapLocationClient(this);
mLocationOption = new AMapLocationClientOption();
//设置定位监听
mlocationClient.setLocationListener(this);
//设置为高精度定位模式
mLocationOption.setLocationMode(AMapLocationClientOption.AMapLocationMode.Hight_Accuracy);
//设置定位参数
mlocationClient.setLocationOption(mLocationOption);
mLocationOption.setOnceLocation(false);
mLocationOption.setGpsFirst(true);
// 设置发送定位请求的时间间隔,最小值为1000ms,1秒更新一次定位信息
mLocationOption.setInterval(10000);
// 此方法为每隔固定时间会发起一次定位请求,为了减少电量消耗或网络流量消耗,
// 注意设置合适的定位时间的间隔(最小间隔支持为2000ms),并且在合适时间调用stopLocation()方法来取消定位请求
// 在定位结束后,在合适的生命周期调用onDestroy()方法
// 在单次定位情况下,定位无论成功与否,都无需调用stopLocation()方法移除请求,定位sdk内部会移除
mLocationOption.setMockEnable(true);
mlocationClient.startLocation();
}
}
}
三、修改混淆文件
(这里不加也可以,但是有可能会出现报错但还能运行或者是闪退的情况)
-
修改bulid.gradle文件
image.png 在下面proguard-rules.pro文件加入
-keep class com.amap.api.location.**{*;}
-keep class com.amap.api.fence.**{*;}
-keep class com.loc.**{*;}
-keep class com.amap.api.services.**{*;}
四、结果
短距离步行效果图.jpg
不知道是不是定位不精确的原因,短距离步行显示的差距有点大,也有可能是第一次定位和第二次定位会出现偏差导致了第一个线段的不准确。
骑着我的小电瓶效果图.jpg
远距离的轨迹跟踪还是比较准确的。