一. LocationManager:
LocationManager系统服务是位置服务的核心组件,它提供了一系列方法来处理与位置相关的问题,比如查询上一个已知位置,定期更新设备的地理位置,或者当设备进入给定地理位置附近时,触发应用指定意图等;
(A). 使用LocationManager需要以下过程;
1.获取LocationManager,它不能直接实例化:
LocationManager lm = (LocationManager) getSystemService(Context.LOCATION_SERVICE);
2.了解LocationProvider:它是位置信息提供者,系统一般提供三种方式获取地理位置信息:
(1)GPS_PROVIDER:通过 GPS 来获取地理位置的经纬度信息;
优点:获取地理位置信息精确度高;
缺点:只能在户外使用,获取经纬度信息耗时,耗电;
(2)NETWORK_PROVIDER:通过移动网络的基站或者 Wi-Fi 来获取地理位置;
优点:只要有网络,就可以快速定位,室内室外都可;
缺点:精确度不高;
(3)PASSIVE_PROVIDER:被动接收更新地理位置信息,而不用自己请求地理位置信息。 PASSIVE_PROVIDER 返回的位置是通过其他 providers 产生的,可以查询 getProvider() 方法决定位置更新的由来,需要 ACCESS_FINE_LOCATION 权限,但是如果未启用 GPS,则此 provider 可能只返回粗略位置匹配;
获取provider的方法有getProviders,getAllProviders,getBestProvider(根据一组条件来返回合适的provider)
List<String> list = locationManager.getProviders(true);
if (list != null) {
for (String x : list) {
Log.e("gzq", "name:" + x);
}
}
LocationProvider lpGps = locationManager.getProvider(LocationManager.GPS_PROVIDER);
LocationProvider lpNet = locationManager.getProvider(LocationManager.NETWORK_PROVIDER);
LocationProvider lpPsv = locationManager.getProvider(LocationManager.PASSIVE_PROVIDER);
Criteria criteria = new Criteria();
// Criteria是一组筛选条件
criteria.setAccuracy(Criteria.ACCURACY_FINE);
//设置定位精准度
criteria.setAltitudeRequired(false);
//是否要求海拔
criteria.setBearingRequired(true);
//是否要求方向
criteria.setCostAllowed(true);
//是否要求收费
criteria.setSpeedRequired(true);
//是否要求速度
criteria.setPowerRequirement(Criteria.NO_REQUIREMENT);
//设置电池耗电要求
criteria.setBearingAccuracy(Criteria.ACCURACY_HIGH);
//设置方向精确度
criteria.setSpeedAccuracy(Criteria.ACCURACY_HIGH);
//设置速度精确度
criteria.setHorizontalAccuracy(Criteria.ACCURACY_HIGH);
//设置水平方向精确度
criteria.setVerticalAccuracy(Criteria.ACCURACY_HIGH);
//设置垂直方向精确度
//返回满足条件的当前设备可用的provider,第二个参数为false时返回当前设备所有provider中最符合条件的那个provider,但是不一定可用
String mProvider = locationManager.getBestProvider(criteria, true);
if (mProvider != null) {
Log.e("gzq", "mProvider:" + mProvider);
}
}
打印出来的结果就是passive,gps,network;
3.声明权限
(1)ACCESS_FINE_LOCATION是精确位置,如果使用GPS_PROVIDER或者同时使用GPS_PROVIDER和NETWORK_PROVIDER,需声明该权限,它对于这两个provider都是有效的;
(2)ACCESS_COARSE_LOCATION是粗略位置,该权限只针对NETWORK_PROVIDER。
4.注册一个位置监听器来接受结果
private final class MyLocationListener implements LocationListener{
public void onLocationChanged(Location location) {
Log.e("gzq", "onLocationChanged" + location.toString());
}
public void onStatusChanged(String provider, int status, Bundle extras) {
Log.e("gzq", "onStatusChanged" + status);
}
public void onProviderEnabled(String provider) {
Log.e("gzq", "onProviderEnabled");
}
public void onProviderDisabled(String provider) {
Log.e("gzq", "onProviderDisabled");
}
}
这个回调里面有4个方法;
(1)onLocationChanged:当位置发生改变后就会回调该方法,经纬度相关信息存在Location里面;
(2)onStatusChanged:我们所采用的provider状态改变时会回调,该状态有3种;
LocationProvider.OUT_OF_SERVICE = 0:无服务
LocationProvider.AVAILABLE = 2:provider可用
LocationProvider.TEMPORARILY_UNAVAILABLE = 1:provider不可用
(3)onProviderEnabled:当provider可用时被触发,比如定位模式切换到了使用精确位置时GPSProvider就会回调该方法;
(4)onProviderDisabled:当provider不可用时被触发,比如定位模式切换到了使用使用网络定位时GPSProvider就会回调该方法;
5.获取位置信息,调用监听方法,不过在获取位置前先判断一下要调用的provider是否可用;
if (mLocationManager.isProviderEnabled(LocationManager.GPS_PROVIDER)) {
locationManager.requestLocationUpdates(LocationManager.GPS_PROVIDER, 5,10, locationListener);
}
这个方法表明要跟踪GPS位置的变化,并且每5秒刷新一次,同时两次的位置的间隔要超过10米;
在不需要位置的时候解注册监听:
locationManager.removeUpdates(locationListener);
关于requestLocationUpdates有一些问题需要注意:
(1)刚才我们传入的是Listener,其实也可以用PendingIntent来代替Listener,当位置更新时会通过广播回调,使用2个键KEY_LOCATION_CHANGED和Location来接收位置变化;
(2)参数这里里可以传递一个Looper,如果不指定的话,则调用线程必须已经是一个Looper线程,比如调用Activity的主线程,如果指定了Looper,则在提供的Looper线程上进行回调;
(B). LocationManager还提供了一些其他有用的方法:
1.获取缓存中的位置信息getLastKnownLocation
Location location = lm.getLastKnownLocation(LocationManager.GPS_PROVIDER);
该方法不会发起监听,返回的是上一次的位置信息,但此前如果没有位置更新的话,返回的位置信息可能是错误的;
2.获取一次定位结果requestSingleUpdate
locationManager.requestSingleUpdate(LocationManager.GPS_PROVIDER,locationListener,null);
如果不想一直监听位置信息,那么可以用requestSingleUpdate来实现只请求一次定位,该方法也要在主线程上执行;
3.发送辅助指令sendExtraCommand
mLocationManager.sendExtraCommand("LOC","NOTIFY_DOWNLOAD", extras);
可以通过Bundle参数来发送相关信息;
4.判断provider是否可用
mLocationManager.isProviderEnabled(LocationManager.GPS_PROVIDER);
5.添加地理围栏
locationManager.addProximityAlert(38.234, 114.234, 5, -1, PendingIntent.getBroadcast(this, 1, new Intent(), 3));
可以设置一个区域,当进入或离开这个区域的时候会收到通知,前两个参数指定一个点,第三个参数是半径,第四个参数是超时时间,设置为-1表示不存在超时,最后一个是广播接收器。
触发的Intent将使用键KEY_PROXIMITY_ENTERING,如果值为true,则设备进入邻近区域,如果是false,说明设备离开该区域。