定位,算是android应用中一个非常常见的功能了吧,并且大部分的应用里基本也都会用到地图,所以以前的做法基本都是直接接入了百度、高德这些第三方定位,也没遇到过什么问题。
然而,最近的工作基本上都是做一些这样那样的小工具,某些功能里的确涉及到定位功能,并且还没有地图的使用,如果这样还是接入百度高德这些未免就有些画蛇添足了,使用gps功能就足够了。
大致看了下LocationManager类的几个方法,也就是常用的几个方法,然后再找篇帖子,大笔一挥,申请权限、打开位置服务、取经纬度,然后。。。卧槽,经纬度呢,怎么没返回??赶紧再去搜几篇帖子看看问题能否找到原因,千篇一律的内容,无外乎使用
LocationManager.getLastKnownLocation(LocationManager.GPS_PROVIDER);
或者
LocationManager.requestLocationUpdates(LocationManager.GPS_PROVIDER,x,x,LocationListener);
然后还是获取不到经纬度,又有人说了,是定位要提前开启,requestLocation要提前调用,然后也照做了,还是不行。网上的帖子真的是复制粘贴居多,最终也没解决问题,算了,还是再看看LocationManager里还有哪些没用到的方法吧。还真就发现了一个,getBestProvider,之前都是自己指定的provider,这个居然名字都这么直白暴力,返回最好的provider,赶紧看了下方法说明
/**
*@param criteriathe criteria that need to be matched
*@param enabledOnly if true then only a provider that is currently enabled is returned
*@returnname of the provider that best matches the requirements
*/
public String getBestProvider(Criteria criteria, booleanenabledOnly) {
checkCriteria(criteria);
try{
returnmService.getBestProvider(criteria,enabledOnly);
} catch (RemoteException e) {
throwe.rethrowFromSystemServer();
}
}
然后就赶紧试了下该方案是否能够成功返回,具体操作还是直接上代码吧,比较直观
public class LocationUtils implements LocationListener {
private static volatile LocationUtils instance;
private CLocationListener mLocationListener;
private LocationManager mLocationManager;
private Context context;
private String mProviderName;
//为避免内存泄漏,此处传入的一定要是application上下文对象
public static LocationUtils getInstance(Context context) {
if (instance == null) {
synchronized (LocationUtils.class) {
if (instance == null) {
instance = new LocationUtils(context);
}
}
}
return instance;
}
public LocationUtils(final Context context) {
this.context = context;
new Handler(Looper.getMainLooper()).post(new Runnable() {
@Override
public void run() {
try {
mLocationManager = (LocationManager) context.getSystemService(Context.LOCATION_SERVICE);
// 查找到服务信息
Criteria criteria = new Criteria();
// 设置定位精确度 Criteria.ACCURACY_COARSE比较粗略,Criteria.ACCURACY_FINE则比较精细
criteria.setAccuracy(Criteria.ACCURACY_FINE);
// 设置是否要求速度
criteria.setSpeedRequired(false);
// 设置是否需要海拔信息
criteria.setAltitudeRequired(false);
// 设置是否需要方位信息
criteria.setBearingRequired(false);
// 设置是否允许运营商收费
criteria.setCostAllowed(true);
// 设置对电源的需求
criteria.setPowerRequirement(Criteria.POWER_LOW); // 低功耗
// 为获取地理位置信息时设置查询条件
mProviderName = mLocationManager.getBestProvider(criteria, true); // 获取GPS信息
if (!TextUtils.isEmpty(mProviderName)) {
mLocationManager.requestLocationUpdates(mProviderName, 1000, 0, LocationUtils.this);
}
} catch (Exception e) {
e.printStackTrace();
}
}
});
}
public static boolean isGpsEnabled(Context context) {
return ((LocationManager) context.getSystemService(Context.LOCATION_SERVICE)).isProviderEnabled(android.location.LocationManager.GPS_PROVIDER);
}
public void requestLocationUpdates(CLocationListener listener) {
mLocationListener = listener;
}
public void removeUpdates() {
mLocationListener = null;
}
@Override
public void onLocationChanged(Location location) {
if (mLocationListener != null) {
mLocationListener.onLocationChanged(location);
}
}
@Override
public void onStatusChanged(String provider, int status, Bundle extras) {
}
@Override
public void onProviderEnabled(String provider) {
Location location = mLocationManager.getLastKnownLocation(provider);
if (mLocationListener != null) {
mLocationListener.onLocationChanged(location);
}
}
@Override
public void onProviderDisabled(String provider) {
}
public interface CLocationListener {
void onLocationChanged(Location location);
}
}
就简单写了几个方法,大致就是这样了,如果需要其他的方法,可以自行根据业务需求进行添加。
不过值得一提的是,位置信息服务的打开,现在的手机基本都是支持三种模式,高准确度模式、低耗电模式、仅限设备模式,前两种因为都支持网络定位,所以即便在室内也能成功返回经纬度,而最后一种仅限设备模式就是仅仅使用设备的gps定位而不使用网络定位,这样的话在室内测试基本不会返回结果,但是在室外开阔地带没有问题(亲测),有需要的朋友可以自行测试。