這篇教程主要介紹了在Android平臺上如何使用服務完成定位功能。眾所周知,Android裝置的當前位置資訊,對開發創新性App、解決人們日常生活問題有極大幫助。在Android平臺開發定位相關的應用程式,需要位置提供者。有兩種型別的位置提供者:
- GPS定位
- 網路定位
以上兩種型別,任何一種都可以獲得使用者或者使用者裝置的位置資訊。但是,它們各有優劣,推薦兩者同時使用。GPS 定位,在室內反應遲緩,比較耗時;網路定位,在沒有網路的時候無法獲得位置資訊。
GPS定位 VS 網路定位
- 獲取位置座標時,網路定位比GPS定位稍快。
- GPS在室內定位非常緩慢,並且比較耗電。
- 網路定位依賴蜂窩網路,獲取的是最近的網路基站的位置。
- GPS定位資料相對精確,得到我們當前的位置資訊。
獲取定位資料
- 在
Manifest
檔案中授權,接收定位資料。 - 建立
LocationManager
例項,將其指向定位服務。 LocationManager
請求定位資料。- 定數資料改變時,
LocationListener
接收更新的定位資料。
授權接收定位更新資料
在Manifest檔案中獲取如下許可權,然後可以通過定位提供者獲得定位資料:
1 2 3 4 5 6 |
; html-script: false ] <manifest ... > <uses-permission android:name="android.permission.ACCESS_FINE_LOCATION" /> <uses-permission android:name="android.permission. ACCESS_COARSE_LOCATION" /> <uses-permission android:name="android.permission.INTERNET" /> </manifest> |
定位提供者必需要有INTERNET
許可權和ACCESS_FINE_LOCATION
許可權。同時,網路定位還需要ACCESS_COARSE _LOCATION
許可權。
建立LocationManager例項,指向定位服務
無論何種型別的Android後臺Service,需要獲得其引用才能使用。同樣,通過getSystemService()
方法獲得定位服務的引用,然後將這個引用將新增到新建立的LocationManager
例項中,示例如下:
1 2 |
; html-script: false ] locationManager = (LocationManager) getSystemService(Context.LOCATION_SERVICE); |
從LocationManager請求當前位置
穿件位置服務引用後,通過LocationManager
的requestLocationUpdates()
方法可以請求位置更新資訊。呼叫方法時,需要位置提供者、最後一次更新距今的時間(秒)、距離和LocationListener
物件。呼叫後LocationListener
物件會根據位置進行更新。
1 2 |
; html-script: false ] locationManager.requestLocationUpdates(LocationManager.GPS_PROVIDER, 0, 0, this); |
通過LocationListener,獲得更新位置資料
根據指定的距離或時間間隔,LocationListener會收到更新通知。
示例:獲取當前位置
這個示例通過GPS定位獲取當前位置資料。主要程式碼如下:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 |
; html-script: false ] package com.javapapers.android.geolocationfinder; import android.os.Bundle; import android.app.Activity; import android.content.Context; import android.location.Location; import android.location.LocationListener; import android.location.LocationManager; import android.widget.TextView; import android.util.Log; public class MainActivity extends Activity implements LocationListener { protected LocationManager locationManager; protected LocationListener locationListener; protected Context context; TextView txtLat; String lat; String provider; protected String latitude, longitude; protected boolean gps_enabled, network_enabled; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); txtLat = (TextView) findViewById(R.id.textview1); locationManager = (LocationManager) getSystemService(Context.LOCATION_SERVICE); locationManager.requestLocationUpdates(LocationManager.GPS_PROVIDER, 0, 0, this); } @Override public void onLocationChanged(Location location) { txtLat = (TextView) findViewById(R.id.textview1); txtLat.setText("Latitude:" + location.getLatitude() + ", Longitude:" + location.getLongitude()); } @Override public void onProviderDisabled(String provider) { Log.d("Latitude", "disable"); } @Override public void onProviderEnabled(String provider) { Log.d("Latitude", "enable"); } @Override public void onStatusChanged(String provider, int status, Bundle extras) { Log.d("Latitude", "status"); } } |
佈局檔案如下,
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 |
; html-script: false ] <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:tools="http://schemas.android.com/tools" android:layout_width="match_parent" android:layout_height="match_parent" tools:context=".MainActivity" > <TextView android:id="@+id/textview1" android:layout_width="wrap_content" android:layout_height="wrap_content" android:layout_centerHorizontal="true" android:layout_centerVertical="true" android:text="@string/hello_world" /> </RelativeLayout> |
Manifest檔案如下,
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 |
; html-script: false ] <?xml version="1.0" encoding="utf-8"?> <manifest xmlns:android="http://schemas.android.com/apk/res/android" package="com.javapapers.android.geolocationfinder" android:versionCode="1" android:versionName="1.0" > <uses-sdk android:minSdkVersion="8" android:targetSdkVersion="17" /> <uses-permission android:name="android.permission.ACCESS_FINE_LOCATION" /> <application android:allowBackup="true" android:icon="@drawable/ic_launcher" android:label="@string/app_name" android:theme="@style/AppBaseTheme" > <activity android:name="com.javapapers.android.geolocationfinder.MainActivity" android:label="@string/app_name" > <intent-filter> <action android:name="android.intent.action.MAIN" /> <category android:name="android.intent.category.LAUNCHER" /> </intent-filter> </activity> </application> </manifest> |
輸出效果
提示:如果使用模擬器執行這個示例,需要將準確的經緯度傳送到模擬器。
如何傳送經緯度到模擬器
- 開啟Eclipse中 DDMS 檢視(Window–>Open Perspective)
- 選擇模擬器
- 選擇模擬器控制選項
- 在位置控制皮膚,選擇手動輸入,新增經緯度資料,點選“傳送”