不說廢話,直接說說實現android定位有關的API吧。
這些API都在android.location包下,一共有三個接口和八個類。它們配合使用即可實現定位功能。
三個接口:
GpsStatus.Listener:這是一個當GPS狀態發生改變時,用來接收通知的接口。
GpsStatus.NmeaListener:這是一個用來從GPS裡接收Nmea-0183(為海用電子設備制定的標準格式)信息的接口。
LocationListener:位置監聽器,用於接收當位置信息發生改變時從LocationManager接收通知的接口。
八個類:
Address:描述地址的類,比如:北京天安門
Criteria:用於描述Location Provider標準的類,標準包括位置精度水平,電量消耗水平,是否獲取海拔、方位信息,是否允許接收付費服務。
GeoCoder:用於處理地理位置的編碼。
GpsSatellite:和GpsStatus聯合使用,用於描述當前GPS衛星的狀態。
GpsStatus:和GpsStatus.Listener聯合使用,用於描述當前GPS衛星的狀態。
Location:用於描述位置信息。
LocationManager:通過此類獲取和調用系統位置服務
LocationProvider:用於描述Location Provider的抽象超類,一個LocationProvider應該能夠周期性的報告當前設備的位置信息。
這裡通過一個示例代碼來演示一下android定位。
首先,在AndroidManifest.xml清單文件裡需要加入ACCESS_FINE_LOCATION權限:
package com.test; import java.io.IOException; import java.util.List; import android.app.Activity; import android.location.Address; import android.location.Criteria; import android.location.Geocoder; import android.location.Location; import android.location.LocationListener; import android.location.LocationManager; import android.os.Bundle; import android.util.Log; import android.widget.Toast; public class PositionActivity extends Activity { @Override public void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.main); // 獲取到LocationManager對象 LocationManager locationManager = (LocationManager) getSystemService(LOCATION_SERVICE); // 創建一個Criteria對象 Criteria criteria = new Criteria(); // 設置粗略精確度 criteria.setAccuracy(Criteria.ACCURACY_COARSE); // 設置是否需要返回海拔信息 criteria.setAltitudeRequired(false); // 設置是否需要返回方位信息 criteria.setBearingRequired(false); // 設置是否允許付費服務 criteria.setCostAllowed(true); // 設置電量消耗等級 criteria.setPowerRequirement(Criteria.POWER_HIGH); // 設置是否需要返回速度信息 criteria.setSpeedRequired(false); // 根據設置的Criteria對象,獲取最符合此標準的provider對象 41 String currentProvider = locationManager .getBestProvider(criteria, true); Log.d("Location", "currentProvider: " + currentProvider); // 根據當前provider對象獲取最後一次位置信息 44 Location currentLocation = locationManager .getLastKnownLocation(currentProvider); // 如果位置信息為null,則請求更新位置信息 46 if (currentLocation == null) { locationManager.requestLocationUpdates(currentProvider, 0, 0, locationListener); } // 直到獲得最後一次位置信息為止,如果未獲得最後一次位置信息,則顯示默認經緯度 50 // 每隔10秒獲取一次位置信息 51 while (true) { currentLocation = locationManager .getLastKnownLocation(currentProvider); if (currentLocation != null) { Log.d("Location", "Latitude: " + currentLocation.getLatitude()); Log.d("Location", "location: " + currentLocation.getLongitude()); break; } else { Log.d("Location", "Latitude: " + 0); Log.d("Location", "location: " + 0); } try { Thread.sleep(10000); } catch (InterruptedException e) { Log.e("Location", e.getMessage()); } } // 解析地址並顯示 69 Geocoder geoCoder = new Geocoder(this); try { int latitude = (int) currentLocation.getLatitude(); int longitude = (int) currentLocation.getLongitude(); List<Address> list = geoCoder.getFromLocation(latitude, longitude, 2); for (int i = 0; i < list.size(); i++) { Address address = list.get(i); Toast.makeText( PositionActivity.this, address.getCountryName() + address.getAdminArea() + address.getFeatureName(), Toast.LENGTH_LONG) .show(); } } catch (IOException e) { Toast.makeText(PositionActivity.this, e.getMessage(), Toast.LENGTH_LONG).show(); } } // 創建位置監聽器 85 private LocationListener locationListener = new LocationListener() { // 位置發生改變時調用 87 @Override public void onLocationChanged(Location location) { Log.d("Location", "onLocationChanged"); Log.d("Location", "onLocationChanged Latitude" + location.getLatitude()); Log.d("Location", "onLocationChanged location" + location.getLongitude()); } // provider失效時調用 95 @Override public void onProviderDisabled(String provider) { Log.d("Location", "onProviderDisabled"); } // provider啟用時調用101 @Override public void onProviderEnabled(String provider) { Log.d("Location", "onProviderEnabled"); } // 狀態改變時調用107 @Override public void onStatusChanged(String provider, int status, Bundle extras) { Log.d("Location", "onStatusChanged"); } }; }
由於代碼裡的Criteria對象對位置精度要求並不高,所以一般會返回“network”作為provider,而基於network的定位往往會存在一定的位置偏差,這對於需要精確定位的應用程序來說,顯然不合要求。這時,需要則需要用到基於GPS的定位方法瞭。具體請接著看文Android定位功能(二)
Android定位功能(二) (轉到第一部分)
在前文Android定位功能(一)中,已經大致介紹瞭一下在Android平臺中,和定位功能相關的類,並舉例獲取瞭位置信息。但是前文是基於Criteria定制瞭一個標準,通過getBestProvider()方法由Android系統自動獲取最符合Criteria的LocationProvider,從而實現瞭定位功能。這樣的做法能最大限度的保證定位功能的可實現性,但是卻無法保證獲取到的位置信息有最大的準確度。因為除瞭GPS外,其他定位方式都或多或少存在著位置偏移。
在實現GPS定位前,先瞭解一下GPS的部分特性:
1:GPS定位需要依靠3顆或以上的衛星。
2:GPS定位受環境影響較大,在晴朗的空地上,較容易搜索到衛星,而在室內通常是無法搜索到衛星的。
3:GPS定位需要使用GPS功能模塊,而GPS功能模塊的耗電量是巨大的。
在Android系統中,實現GPS定位的思路大致是:
1、獲取GPS的Location Provider。
2、將此Provider傳入到requestLocationUpdates()方法,讓Android系統獲知搜索位置方式。
3、創建實現瞭GpsStatus.Listener接口的對象,重寫onGpsStatusChanged()方法,向LocationManager添加次監聽器,檢測衛星狀態。(可選步驟)
根據以上思路,仿照Android定位功能(一)中的例子,可以很容易的得到以下實現代碼:(此代碼的實現前提是GPS功能模塊處於打開狀態)
public class MainActivity extends Activity { private LocationManager locationManager; private GpsStatus gpsstatus; @Override public void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.main); // 獲取到LocationManager對象 locationManager = (LocationManager) getSystemService(LOCATION_SERVICE); // 根據設置的Criteria對象,獲取最符合此標準的provider對象 String currentProvider = locationManager.getProvider( LocationManager.GPS_PROVIDER).getName(); // 根據當前provider對象獲取最後一次位置信息 Location currentLocation = locationManager .getLastKnownLocation(currentProvider); // 如果位置信息為null,則請求更新位置信息 if (currentLocation == null) { locationManager.requestLocationUpdates(currentProvider, 0, 0, locationListener); } // 增加GPS狀態監聽器 locationManager.addGpsStatusListener(gpsListener); // 直到獲得最後一次位置信息為止,如果未獲得最後一次位置信息,則顯示默認經緯度 // 每隔10秒獲取一次位置信息 while (true) { currentLocation = locationManager .getLastKnownLocation(currentProvider); if (currentLocation != null) { Log.d("Location", "Latitude: " + currentLocation.getLatitude()); Log.d("Location", "location: " + currentLocation.getLongitude()); break; } else { Log.d("Location", "Latitude: " + 0); Log.d("Location", "location: " + 0); } try { Thread.sleep(10000); } catch (InterruptedException e) { Log.e("Location", e.getMessage()); } } } private GpsStatus.Listener gpsListener = new GpsStatus.Listener() { // GPS狀態發生變化時觸發 @Override public void onGpsStatusChanged(int event) { // 獲取當前狀態 gpsstatus = locationManager.getGpsStatus(null); switch (event) { // 第一次定位時的事件 case GpsStatus.GPS_EVENT_FIRST_FIX: break; // 開始定位的事件 case GpsStatus.GPS_EVENT_STARTED: break; // 發送GPS衛星狀態事件 case GpsStatus.GPS_EVENT_SATELLITE_STATUS: Toast.makeText(MainActivity.this,"GPS_EVENT_SATELLITE_STATUS", Toast.LENGTH_SHORT).show(); Iterable<GpsSatellite> allSatellites = gpsstatus .getSatellites(); Iterator<GpsSatellite> it = allSatellites.iterator(); int count = 0; while (it.hasNext()) { count++; } Toast.makeText(MainActivity.this, "Satellite Count:" + count, Toast.LENGTH_SHORT).show(); break; // 停止定位事件 case GpsStatus.GPS_EVENT_STOPPED: Log.d("Location", "GPS_EVENT_STOPPED"); break; } } }; // 創建位置監聽器 private LocationListener locationListener = new LocationListener() { // 位置發生改變時調用 @Override public void onLocationChanged(Location location) { Log.d("Location", "onLocationChanged"); } // provider失效時調用 @Override public void onProviderDisabled(String provider) { Log.d("Location", "onProviderDisabled"); } // provider啟用時調用 @Override public void onProviderEnabled(String provider) { Log.d("Location", "onProviderEnabled"); } // 狀態改變時調用 @Override public void onStatusChanged(String provider, int status, Bundle extras) { Log.d("Location", "onStatusChanged"); } }; }
通過以上代碼中的註釋部分,可以清晰的知道Android定位功能裡相關方法的具體含義。希望對大傢有用。
另外,因為GPS的自身特性,此代碼在室內幾乎無法定位,所以建議再真正的實際項目裡,至少使用network和GPS兩種不同的Location Provider實現定位功能。
本人暫時未找到同時關閉網絡和GPS功能實現定位的方法,本人也未找到通過代碼在沒有ROOT的前提下直接代開網絡和GPS功能的代碼。如果大傢在這兩方面有自己的體會,請不吝賜教,留言評論或給出參考地址都可。大傢一同探討,一同進步。