上次介紹了位置服務中的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”視圖,即可看到一下界面:
如果設置了模擬座標後,在模擬器的狀態欄就會出現一個雷達圖形的標誌,如圖: