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!

推薦閱讀: