1. 程式人生 > >徹底解決Android GPS沒法定位這一頑固問題

徹底解決Android GPS沒法定位這一頑固問題

ive 解決 lan 搜集 final inf 密鑰 基站 問題

大家去網上搜索Android定位location為null沒法定位問題。預計有一大堆文章介紹怎樣來解決。可是最後大家發現基本沒用。

本文將從Android定位實現原理來深入分析沒法定位原因並提出真正的解決方式。

在分析之前,我們肯定得先看看android官方提供的定位SDK。


默認Android GPS定位實例


獲取LocationManager:

mLocationManager = (LocationManager) getSystemService(Context.LOCATION_SERVICE);

選擇Location Provider:

Android系統存在多種provider,各自是

GPS_PROVIDER:

這個就是手機裏有GPS芯片,然後利用該芯片就能利用衛星獲得自己的位置信息。可是在室內,GPS定位基本沒用,非常難定位的到。

NETWORK_PROVIDER:

這個就是利用網絡定位,一般是利用手機基站和WIFI節點的地址來大致定位位置,

這樣的定位方式取決於server,即取決於將基站或WIF節點信息翻譯成位置信息的server的能力。因為眼下大部分Android手機沒有安裝google官方的location manager庫。大陸網絡也不同意。即沒有server來做這個事情,自然該方法基本上沒法實現定位。

PASSIVE_PROVIDER:

被動定位方式。這個意思也比較明顯。就是用現成的,當其它應用使用定位更新了定位信息。系統會保存下來。該應用接收到消息後直接讀取就能夠了。比方假設系統中已經安裝了百度地圖,高德地圖(室內能夠實現精確定位)。你僅僅要使用它們定位過後。再使用這樣的方法在你的程序肯定是能夠拿到比較精確的定位信息。

用戶能夠直接指定某一個provider

String provider = mLocationManager.getProvider(LocationManager.GPS_PROVIDER);

也能夠提供配置,由系統依據用戶的配置為用戶選擇一個最接近用戶需求的provider

Criteria crite = new Criteria();  
crite.setAccuracy(Crite.ACCURACY_FINE); //精度
crite.setPowerRequirement(Crite.POWER_LOW); //功耗類型選擇
String provider = mLocationManager.getBestProvider(crite, true); 

獲取Location

Location location = mLocationManager.getLocation(provider);  

然後你會發現,這個返回的location永遠為null,你自然沒法定位。然後網上到處是咨詢為啥獲得的location為null,相同網絡到處是解決問題的所謂解決方式。

所謂解決方式


網上有人說。一開始location是非常有可能是null的,這是由於程序還從來沒有請求 過,僅僅需又一次請求更新location,並註冊監聽器以接收更新後的location信息。

LocationListener locationListener = new LocationListener() {
        @Override
        public void onStatusChanged(String provider, int status, Bundle extras) {
        }
        @Override
        public void onProviderEnabled(String provider) {
        }

        @Override
        public void onProviderDisabled(String provider) {
        }

        @Override
        public void onLocationChanged(Location location) {
            longitude = location.getLongitude();
            latitude  = location.getLatitude();
            Log.d(TAG,"Location longitude:"+ longitude +" latitude: "+ latitude );
        }
};
mLocationManager.requestLocationUpdates(serviceProvider, 10000, 1, this);

然後你發現onLocationChanged永遠不會被調用,你仍然沒法獲取定位信息。

為什麽就沒法獲取到location呢?


事實上在上面我已經提到了,全部上面的解決的方案都沒有解決根本問題,那就是當你在室內開發時。你的手機根本就沒法獲取位置信息,你叫系統怎樣將位置信息通知給你的程序。

所以要從根本上解決問題,就要解決位置信息獲取問題。剛剛也提到了,僅僅有NETWORK_PROVIDER這樣的模式才是室內定位可靠的方式,僅僅只是因為大陸的怪怪網絡,且大部分廠商也不會用google的服務,這樣的定位方式默認是沒法用的。那怎麽辦?好辦,找個替代的服務商就能夠了,百度的位置信息sdk就能夠解決問題。

它的基本原理在上面已經提到過了,就是搜集你的wifi節點信息和你的手機基站信息來定位。


真正的解決方式,使用百度位置定位SDK


SDK下載:

http://pan.baidu.com/s/1i3xGMih

當然大家能夠在官網下載,這樣能夠下載到最新的sdk

http://lbsyun.baidu.com/sdk/download

SDK使用:

1. 申請百度的服務密鑰。詳細操作步驟見官網:

http://api.map.baidu.com/lbsapi/cloud/geosdk.htm

2.將上面下載的sdk文件locSDK_4.1.jar復制到你項目的libs下

3. 改動AndroidManifest文件,在該文件中加入例如以下配置

        <service
            android:name="com.baidu.location.f"
            android:enabled="true"
            android:process=":remote" >
        </service>
        <meta-data
            android:name="com.baidu.lbsapi.API_KEY"
            android:value="xxxxx " />
	<uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" />
	<uses-permission android:name="android.permission.ACCESS_COARSE_LOCATION" />
	<uses-permission android:name="android.permission.ACCESS_FINE_LOCATION" />

上面meta-data中value的值改為你自己的密鑰

代碼裏調用sdk:

public class LocationUtil {
	private final static boolean DEBUG = true;
	private final static String TAG = "LocationUtil";
	private static LocationUtil mInstance;
	private BDLocation mLocation = null;
	private MLocation  mBaseLocation = new MLocation();

	public static LocationUtil getInstance(Context context) {
		if (mInstance == null) {
			mInstance = new LocationUtil(context);
		}
		return mInstance;
	}

	Context mContext;
	String mProvider;
	public BDLocationListener myListener = new MyLocationListener();
	private LocationClient mLocationClient;
	
	public LocationUtil(Context context) {
	    mLocationClient = new LocationClient(context.getApplicationContext());
		initParams();
		mLocationClient.registerLocationListener(myListener);
	}

	public void startMonitor() {
		if (DEBUG) Log.d(TAG, "start monitor location");
		if (!mLocationClient.isStarted()) {
			mLocationClient.start();
		}
		if (mLocationClient != null && mLocationClient.isStarted()) {
			mLocationClient.requestLocation();
		} else {
			 Log.d("LocSDK3", "locClient is null or not started");
		}
	}

	public void stopMonitor() {
		if (DEBUG) Log.d(TAG, "stop monitor location");
		if (mLocationClient != null && mLocationClient.isStarted()) {
			mLocationClient.stop();
		}
	}
	
	public BDLocation getLocation() {
		if (DEBUG) Log.d(TAG, "get location");
		return mLocation;
	}
	
	public MLocation getBaseLocation() {
		if (DEBUG) Log.d(TAG, "get location");
		return mBaseLocation;
	}
	
	private void initParams() {
		LocationClientOption option = new LocationClientOption();
		option.setOpenGps(true);
		//option.setPriority(LocationClientOption.NetWorkFirst);
		option.setAddrType("all");//返回的定位結果包括地址信息
		option.setCoorType("bd09ll");//返回的定位結果是百度經緯度,默認值gcj02
		option.setScanSpan(5000);//設置發起定位請求的間隔時間為5000ms
		option.disableCache(true);//禁止啟用緩存定位
		option.setPoiNumber(5);    //最多返回POI個數   
		option.setPoiDistance(1000); //poi查詢距離        
		option.setPoiExtraInfo(true); //是否須要POI的電話和地址等具體信息        
		mLocationClient.setLocOption(option);
	}


	public class MyLocationListener implements BDLocationListener {
		@Override
		public void onReceiveLocation(BDLocation location) {
			if (location == null) {
				return ;
			}
			mLocation = location;
			mBaseLocation.latitude = mLocation.getLatitude();
			mBaseLocation.longitude = mLocation.getLongitude();
			
			StringBuffer sb = new StringBuffer(256);
			sb.append("time : ");
			sb.append(location.getTime());
			sb.append("\nerror code : ");
			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());
			sb.append("\ncity : ");
			sb.append(location.getCity());
			if (location.getLocType() == BDLocation.TypeGpsLocation){
				sb.append("\nspeed : ");
				sb.append(location.getSpeed());
				sb.append("\nsatellite : ");
				sb.append(location.getSatelliteNumber());
			} else if (location.getLocType() == BDLocation.TypeNetWorkLocation){
				sb.append("\naddr : ");
				sb.append(location.getAddrStr());
			}
			if (DEBUG) Log.d(TAG, "" + sb);
		}

		public void onReceivePoi(BDLocation poiLocation) {
		}
	}
	
	public class MLocation {
		public double latitude;
		public double longitude;
	}
}

當然別忘了在setting裏將gps定位打開

/********************************

* 本文來自博客 “愛踢門”

* 轉載請標明出處:http://blog.csdn.net/itleaks

******************************************/

徹底解決Android GPS沒法定位這一頑固問題