定位,算是android應(yīng)用中一個非常常見的功能了吧,并且大部分的應(yīng)用里基本也都會用到地圖,所以以前的做法基本都是直接接入了百度、高德這些第三方定位,也沒遇到過什么問題。
然而,最近的工作基本上都是做一些這樣那樣的小工具,某些功能里的確涉及到定位功能,并且還沒有地圖的使用,如果這樣還是接入百度高德這些未免就有些畫蛇添足了,使用gps功能就足夠了。
大致看了下LocationManager類的幾個方法,也就是常用的幾個方法,然后再找篇帖子,大筆一揮,申請權(quán)限、打開位置服務(wù)、取經(jīng)緯度,然后。。。臥槽,經(jīng)緯度呢,怎么沒返回??趕緊再去搜幾篇帖子看看問題能否找到原因,千篇一律的內(nèi)容,無外乎使用
LocationManager.getLastKnownLocation(LocationManager.GPS_PROVIDER);
或者
LocationManager.requestLocationUpdates(LocationManager.GPS_PROVIDER,x,x,LocationListener);
然后還是獲取不到經(jīng)緯度,又有人說了,是定位要提前開啟,requestLocation要提前調(diào)用,然后也照做了,還是不行。網(wǎng)上的帖子真的是復(fù)制粘貼居多,最終也沒解決問題,算了,還是再看看LocationManager里還有哪些沒用到的方法吧。還真就發(fā)現(xiàn)了一個,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;
//為避免內(nèi)存泄漏,此處傳入的一定要是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);
// 查找到服務(wù)信息
Criteria criteria = new Criteria();
// 設(shè)置定位精確度 Criteria.ACCURACY_COARSE比較粗略,Criteria.ACCURACY_FINE則比較精細(xì)
criteria.setAccuracy(Criteria.ACCURACY_FINE);
// 設(shè)置是否要求速度
criteria.setSpeedRequired(false);
// 設(shè)置是否需要海拔信息
criteria.setAltitudeRequired(false);
// 設(shè)置是否需要方位信息
criteria.setBearingRequired(false);
// 設(shè)置是否允許運營商收費
criteria.setCostAllowed(true);
// 設(shè)置對電源的需求
criteria.setPowerRequirement(Criteria.POWER_LOW); // 低功耗
// 為獲取地理位置信息時設(shè)置查詢條件
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);
}
}
就簡單寫了幾個方法,大致就是這樣了,如果需要其他的方法,可以自行根據(jù)業(yè)務(wù)需求進(jìn)行添加。
不過值得一提的是,位置信息服務(wù)的打開,現(xiàn)在的手機基本都是支持三種模式,高準(zhǔn)確度模式、低耗電模式、僅限設(shè)備模式,前兩種因為都支持網(wǎng)絡(luò)定位,所以即便在室內(nèi)也能成功返回經(jīng)緯度,而最后一種僅限設(shè)備模式就是僅僅使用設(shè)備的gps定位而不使用網(wǎng)絡(luò)定位,這樣的話在室內(nèi)測試基本不會返回結(jié)果,但是在室外開闊地帶沒有問題(親測),有需要的朋友可以自行測試。