當(dāng)前位置:首頁(yè) > 嵌入式培訓(xùn) > 嵌入式學(xué)習(xí) > 講師博文 > Android GPS定位步驟
為確定設(shè)備所在位置,需要進(jìn)行以下步驟的操作。
提供位置服務(wù),需要獲得LocationManager對(duì)象。使用LOCATION_SERVICE參數(shù)調(diào)用android.app.Activity.getSystemService()函數(shù)獲取一個(gè)LocationManager實(shí)例。
android.app.Activity.getSystemService()函數(shù)代碼如代碼所示。
getSystemService()
//Context.LOCATION_SERVICE指明獲取的服務(wù)是位置服務(wù)
String serviceString = Context.LOCATION_SERVICE;
//根據(jù)服務(wù)名稱獲取Android提供的系統(tǒng)級(jí)服務(wù)
LocationManagerLocationManager =
(LocationManager)getSystemService(serviceString);
這里介紹一下Android支持的系統(tǒng)級(jí)服務(wù)有哪些,如表所示。
Android支持的系統(tǒng)級(jí)服務(wù)表
Context類的靜態(tài)常量 | 值 | 返回對(duì)象 | 說(shuō)明 |
LOCATION_SERVICE | location | LocationManager | 控制位置等設(shè)備的更新 |
WINDOW_SERVICE | window | WindowManager | 頂層的窗口管理器 |
LAYOUT_INFLATER_SERVICE | layout_inflater | LayoutInflater | 將XML資源實(shí)例化為View |
POWER_SERVICE | power | PowerManager | 電源管理 |
ALARM_SERVICE | alarm | AlarmManager | 在指定時(shí)間接受Intent |
NOTIFICATION_SERVICE | notification | NotificationManager | 后臺(tái)事件通知 |
KEYGUARD_SERVICE | keyguard | KeyguardManager | 鎖定或解鎖鍵盤 |
SEARCH_SERVICE | search | SearchManager | 訪問(wèn)系統(tǒng)的搜索服務(wù) |
VIBRATOR_SERVICE | vibrator | Vibrator | 訪問(wèn)支持振動(dòng)的硬件 |
CONNECTIVITY_SERVICE | connection | ConnectivityManager | 網(wǎng)絡(luò)連接管理 |
WIFI_SERVICE | wifi | WifiManager | WiFi連接管理 |
INPUT_METHOD_SERVICE | input_method | InputMethodManager | 輸入法管理 |
選擇LocationManager的定位方法。在獲取到LocationManager后,還需要指定LocationManager的定位方法,然后才能夠調(diào)用LocationManager,LocationManager支持的定位方法有以下兩種。
GPS定位:可以提供更加精確的位置信息,但定位速度和質(zhì)量受到衛(wèi)星數(shù)量和環(huán)境情況的影響。
網(wǎng)絡(luò)定位:提供的位置信息精度差,但速度較GPS定位快。
LocationManager支持的定位方法比較,如表所示。
LocationManager支持的定位方法比較
LocationManager類的靜態(tài)常量 | 值 | 說(shuō)明 |
GPS_PROVIDER |
gps
|
使用GPS定位,利用衛(wèi)星提供精確的位置信息,需要android.permissions.ACCESS_FINE_LOCATION用戶權(quán)限 |
NETWORK_PROVIDER | network |
使用網(wǎng)絡(luò)定位,利用基站或WiFi提供近似的位置信息,需要具有如下權(quán)限: android.permission.ACCESS_COARSE_LOCATION或android.permission.ACCESS_FINE_LOCATION. |
在指定LocationManager的定位方法后,則可以調(diào)用getLastKnowLocation()方法獲取當(dāng)前的位置信息。
以使用GPS定位為例,獲取位置信息的代碼如代碼清單所示。
代碼清單GPS定位獲取位置信息
String provider = LocationManager.GPS_PROVIDER;
Location location = locationManager.getLastKnownLocation(provider);
在上述代碼中,第2行返回的Location對(duì)象中,包含了可以確定位置的信息,如經(jīng)度、緯度和速度等。然后通過(guò)調(diào)用Location中的getLatitude()和getLongitude()方法可以分別獲取位置信息中的緯度和經(jīng)度,示例代碼如代碼清單所示。
代碼清單獲取經(jīng)緯度信息
doublelat = location.getLatitude();
doublelng = location.getLongitude();
實(shí)現(xiàn)LocationListener類。代碼如代碼清單所示。
代碼清單LocationListener
LocationListenerlocationListener = new LocationListener(){
//在設(shè)備的位置改變時(shí)被調(diào)用
public void onLocationChanged(Location location) {
}
//在用戶禁用具有定位功能的硬件時(shí)被調(diào)用
public void onProviderDisabled(String provider) {
}
//在用戶啟用具有定位功能的硬件時(shí)被調(diào)用
public void onProviderEnabled(String provider) {
}
//在提供定位功能的硬件的狀態(tài)改變時(shí)被調(diào)用,如從不可獲取位置信息狀態(tài)到可以獲取位置信息的狀態(tài),反之亦然
public void onStatusChanged(String provider, int status, Bundle extras) {
}
};
利用requestLocationUpdates()方法啟動(dòng)位置信息的接收。
LocationManager提供了一種便捷、高效的位置監(jiān)視方法requestLocationUpdates(),可以根據(jù)位置的距離變化和時(shí)間間隔設(shè)定產(chǎn)生位置改變事件的條件,這樣可以避免因微小的距離變化而產(chǎn)生大量的位置改變事件。
LocationManager中設(shè)定監(jiān)聽位置變化的代碼如代碼清單所示。
代碼清單監(jiān)聽位置變化
locationManager.requestLocationUpdates(provider, 2000, 10, locationListener);
其中,第1個(gè)參數(shù)是定位的方法,GPS定位或網(wǎng)絡(luò)定位;第2個(gè)參數(shù)是產(chǎn)生位置改變事件的時(shí)間間隔,單位為微秒;第3個(gè)參數(shù)是距離條件,單位是米;第4個(gè)參數(shù)是回調(diào)函數(shù),在滿足條件后的位置改變事件的處理函數(shù)。代碼將產(chǎn)生位置改變事件的條件設(shè)定為距離改變10米,時(shí)間間隔為2秒。
為了使GPS定位功能生效,還需要在AndroidManifest.xml文件中加入用戶許可。沒(méi)有這些權(quán)限,應(yīng)用程序在運(yùn)行時(shí)是無(wú)法獲取到位置更新的。實(shí)現(xiàn)代碼如下所示:
代碼清單 AndroidManifest.xml
<uses-permissionandroid:name="android.permission.ACCESS_FINE_LOCATION"/>
此處需注意,若使用GPS_PROVIDER,則使用ACCESS_FINE_LOCATION權(quán)限;若使用NETWORK_PROVIDER,則使用ACCESS_FINE_LOCATION和ACCESS_COARSE_
LOCATION權(quán)限。
位置服務(wù)一般都需要使用設(shè)備上的硬件,理想的調(diào)試方式是將程序上傳到物理設(shè)備上運(yùn)行,但在沒(méi)有物理設(shè)備的情況下,也可以使用Android模擬器提供的虛擬方式模擬設(shè)備的位置變化,調(diào)試具有位置服務(wù)的應(yīng)用程序。
首先打開DDMS中的模擬器控制,在Location Controls中的Longitude和Latitude部分輸入設(shè)備當(dāng)前的經(jīng)度和緯度,然后單擊“Send”按鈕,就將虛擬的位置信息發(fā)送到Android模擬器中,如圖所示。
在程序運(yùn)行過(guò)程中,可以在模擬器控制器中改變經(jīng)度和緯度坐標(biāo)值,程序在檢測(cè)到位置的變化后,會(huì)將新的位置信息顯示在界面上。
CurrentLocationDemo是一個(gè)提供位置服務(wù)的基本示例,提供了顯示當(dāng)前位置新的功能,并能夠監(jiān)視設(shè)備的位置變化。
CurrentLocationDemo示例中LocationBasedServiceDemo.java文件的完整代碼如代碼清單所示。
代碼清單 LocationBasedServiceDemo.java
packagecn.com.farsight.LocationBasedServiceDemo;
importandroid.app.Activity;
importandroid.content.Context;
importandroid.os.Bundle;
importandroid.widget.TextView;
importandroid.location.Location;
importandroid.location.LocationListener;
importandroid.location.LocationManager;
public class LocationBasedServiceDemo extends Activity {
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.main);
String serviceString = Context.LOCATION_SERVICE;
LocationManagerlocationManager =
(LocationManager)getSystemService(serviceString);
String provider = LocationManager.GPS_PROVIDER;
Location location = locationManager.getLastKnownLocation(provider);
getLocationInfo(location);
locationManager.requestLocationUpdates(provider, 2000, 0, locationListener);
}
private void getLocationInfo(Location location){
String latLongInfo;
TextViewlocationText = (TextView)findViewById(R.id.txtshow);
if (location != null){
doublelat = location.getLatitude();
doublelng = location.getLongitude();
latLongInfo = "Lat: " + lat + "\nLong: " + lng;
}
else{
latLongInfo = "No location found";
}
locationText.setText(“Your Current Position is:\n" + latLongInfo);
}
private final LocationListenerlocationListener = new LocationListener(){
@Override
public void onLocationChanged(Location location) {
getLocationInfo(location);
}
@Override
public void onProviderDisabled(String provider) {
getLocationInfo(null);
}
@Override
public void onProviderEnabled(String provider) {
getLocationInfo(null);
}
@Override
public void onStatusChanged(String provider, int status, Bundle extras) {
}
};
}