Android Location服務之LocationManager案例詳解
上次介紹瞭位置服務中的Geocoder,這次就來介紹一下LocationManager。LocationManager系統服務是位置服務的核心組件,它提供瞭一系列方法來處理與位置相關的問題,包括查詢上一個已知位置、註冊和註銷來自某個LocationProvider的周期性的位置更新、註冊和註銷接近某個坐標時對一個已定義的Intent的觸發等。今天我們就一起探討一下LocationManager的簡單應用。
在進入正題之前,朋友們需要瞭解與LocationManager相關的兩個知識點:
provider:LocationManager獲取位置信息的途徑,常用的有兩種:GPS和NETWORK。GPS定位更精確,缺點是隻能在戶外使用,耗電嚴重,並且返回用戶位置信息的速度遠不能滿足用戶需求。NETWORK通過基站和Wi-Fi信號來獲取位置信息,室內室外均可用,速度更快,耗電更少。為瞭獲取用戶位置信息,我們可以使用其中一個,也可以同時使用兩個。
LocationListener:位置監聽器接口,定義瞭常見的provider狀態變化和位置的變化的方法,我們需要實現此接口,完成自己的處理邏輯,然後讓LocationManager註冊此監聽器,完成對各種狀態的監聽。
既然上面講到位置服務的核心是LocationManager,那麼我們如何取得一個LocationManager呢?像其他系統服務一樣,通過以下方式即可得到一個LocationManager實例:
LocationManager locMgr = (LocationManager) getSystemService(Context.LOCATION_SERVICE);
對象實例是獲取到瞭,可是怎麼應用呢?下面就通過一個示例具體演示一下。
我們新建一個location項目。因為示例是基於地圖服務的,所以創建時別忘瞭Build Target要選中Google APIs這一項。
然後修改/res/layout/main.xml,代碼如下:
<?xml version="1.0" encoding="utf-8"?> <FrameLayout xmlns:android="http://schemas.android.com/apk/res/android" android:orientation="vertical" android:layout_width="fill_parent" android:layout_height="fill_parent"> <com.google.android.maps.MapView android:id="@+id/mapView" android:layout_width="fill_parent" android:layout_height="fill_parent" android:clickable="true" android:apiKey="your apiKey goes here"/> <Button android:id="@+id/removeUpdates" android:layout_width="fill_parent" android:layout_height="wrap_content" android:text="removeUpdates"/> </FrameLayout>
然後我們來看以下MainActivity.java文件,代碼如下:
package com.scott.location; import android.content.Context; import android.location.Location; import android.location.LocationListener; import android.location.LocationManager; import android.os.Bundle; import android.view.View; import android.widget.Button; import android.widget.ImageView; import android.widget.Toast; import com.google.android.maps.GeoPoint; import com.google.android.maps.MapActivity; import com.google.android.maps.MapView; import com.google.android.maps.MapView.LayoutParams; public class MainActivity extends MapActivity { private MapView mapView; @Override public void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.main); mapView = (MapView) findViewById(R.id.mapView); mapView.getController().setZoom(17); final LocationManager locMgr = (LocationManager) getSystemService(Context.LOCATION_SERVICE); //獲取緩存中的位置信息 Location location = locMgr.getLastKnownLocation(LocationManager.GPS_PROVIDER); if (location != null) { markCurrLocation(location); } final MyLocationListener listener = new MyLocationListener(); //註冊位置更新監聽(最小時間間隔為5秒,最小距離間隔為5米) locMgr.requestLocationUpdates(LocationManager.GPS_PROVIDER, 5000, 5, listener); Button removeUpdates = (Button) findViewById(R.id.removeUpdates); removeUpdates.setOnClickListener(new View.OnClickListener() { @Override public void onClick(View v) { //停止監聽 locMgr.removeUpdates(listener); } }); } /** * 標記當前位置 * @param location */ private void markCurrLocation(Location location) { mapView.removeAllViews(); //清除地圖上所有標記視圖 GeoPoint point = new GeoPoint((int) (location.getLatitude() * 1E6), (int) (location.getLongitude() * 1E6)); mapView.getController().animateTo(point); final MapView.LayoutParams params = new MapView.LayoutParams(LayoutParams.WRAP_CONTENT, LayoutParams.WRAP_CONTENT, point, LayoutParams.BOTTOM_CENTER); final ImageView marker = new ImageView(MainActivity.this); marker.setImageResource(R.drawable.marker); marker.setOnClickListener(new View.OnClickListener() { @Override public void onClick(View v) { Toast.makeText(getApplicationContext(), "hello, location manager!", Toast.LENGTH_SHORT).show(); } }); mapView.addView(marker, params); } @Override protected boolean isRouteDisplayed() { return false; } private final class MyLocationListener implements LocationListener { @Override public void onLocationChanged(Location location) { markCurrLocation(location); } @Override public void onStatusChanged(String provider, int status, Bundle extras) { //Provider狀態在可用、暫不可用、無服務三個狀態之間直接切換時觸發此函數 } @Override public void onProviderEnabled(String provider) { //Provider被enable時觸發此函數,比如GPS被打開 } @Override public void onProviderDisabled(String provider) { //Provider被disable時觸發此函數,比如GPS被關閉 } } }
因為用到瞭地圖服務,所以需要在AndroidManifest.xml中的application標簽之間加入google map library聲明:
<uses-library android:name="com.google.android.maps" />
然後加入位置服務所需的權限:
<uses-permission android:name="android.permission.ACCESS_FINE_LOCATION" />
這裡朋友們需要註意:如果使用GPS_PROVIDER或者同時使用GPS_PROVIDER和NETWORK_PROVIDER,則隻需聲明ACCESS_FINE_LOCATION權限,它對於上述兩個provider都是有效的;而ACCESS_COARSE_LOCATION權限隻針對NETWORK_PROVIDER。
如果是在模擬器裡調試的話,我們可以用以下兩種方法設置一個模擬的坐標值:geo命令和DDMS。
先來說一下geo命令,它需要telnet到本機的5554端口,然後在命令行下輸入geo fix命令,參數可附帶經度、緯度和海拔(可選)。
具體操作如圖:
如果朋友用的系統是windows7的話,會遇到一些小小的麻煩,因為windows7默認是沒有裝Telnet服務,所以我們需要手動安裝一下,點擊“開始->控制面板->程序->程序和功能”,然後再彈出的窗口左側點擊“打開或關閉Windows功能”,會彈出一下界面,選中“Telnet客戶端”和“Telnet服務端”即可。如圖:
不過,使用geo命令還是挺麻煩的,ADT提供瞭一個設置模擬坐標的界面,打開“Emulator Control”視圖,即可看到一下界面:
如果設置瞭模擬坐標後,在模擬器的狀態欄就會出現一個雷達圖形的標志,如圖:
到此這篇關於Android Location服務之LocationManager案例詳解的文章就介紹到這瞭,更多相關Android Location服務之LocationManager內容請搜索WalkonNet以前的文章或繼續瀏覽下面的相關文章希望大傢以後多多支持WalkonNet!
推薦閱讀:
- Android原生定位服務LocationManager
- Android 模擬地圖定位功能的實現
- Android實現簡單計算器界面
- Android實現文字消除效果
- Android ViewStub使用方法學習