Android Location服務(wù)之LocationManager案例詳解
上次介紹了位置服務(wù)中的Geocoder,這次就來(lái)介紹一下LocationManager。LocationManager系統(tǒng)服務(wù)是位置服務(wù)的核心組件,它提供了一系列方法來(lái)處理與位置相關(guān)的問(wèn)題,包括查詢上一個(gè)已知位置、注冊(cè)和注銷來(lái)自某個(gè)LocationProvider的周期性的位置更新、注冊(cè)和注銷接近某個(gè)坐標(biāo)時(shí)對(duì)一個(gè)已定義的Intent的觸發(fā)等。今天我們就一起探討一下LocationManager的簡(jiǎn)單應(yīng)用。
在進(jìn)入正題之前,朋友們需要了解與LocationManager相關(guān)的兩個(gè)知識(shí)點(diǎn):
provider:LocationManager獲取位置信息的途徑,常用的有兩種:GPS和NETWORK。GPS定位更精確,缺點(diǎn)是只能在戶外使用,耗電嚴(yán)重,并且返回用戶位置信息的速度遠(yuǎn)不能滿足用戶需求。NETWORK通過(guò)基站和Wi-Fi信號(hào)來(lái)獲取位置信息,室內(nèi)室外均可用,速度更快,耗電更少。為了獲取用戶位置信息,我們可以使用其中一個(gè),也可以同時(shí)使用兩個(gè)。
LocationListener:位置監(jiān)聽(tīng)器接口,定義了常見(jiàn)的provider狀態(tài)變化和位置的變化的方法,我們需要實(shí)現(xiàn)此接口,完成自己的處理邏輯,然后讓LocationManager注冊(cè)此監(jiān)聽(tīng)器,完成對(duì)各種狀態(tài)的監(jiān)聽(tīng)。
既然上面講到位置服務(wù)的核心是LocationManager,那么我們?nèi)绾稳〉靡粋€(gè)LocationManager呢?像其他系統(tǒng)服務(wù)一樣,通過(guò)以下方式即可得到一個(gè)LocationManager實(shí)例:
LocationManager locMgr = (LocationManager) getSystemService(Context.LOCATION_SERVICE);
對(duì)象實(shí)例是獲取到了,可是怎么應(yīng)用呢?下面就通過(guò)一個(gè)示例具體演示一下。
我們新建一個(gè)location項(xiàng)目。因?yàn)槭纠腔诘貓D服務(wù)的,所以創(chuàng)建時(shí)別忘了Build Target要選中Google APIs這一項(xiàng)。
然后修改/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>
然后我們來(lái)看以下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();
//注冊(cè)位置更新監(jiān)聽(tīng)(最小時(shí)間間隔為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) {
//停止監(jiān)聽(tīng)
locMgr.removeUpdates(listener);
}
});
}
/**
* 標(biāo)記當(dāng)前位置
* @param location
*/
private void markCurrLocation(Location location) {
mapView.removeAllViews(); //清除地圖上所有標(biāo)記視圖
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狀態(tài)在可用、暫不可用、無(wú)服務(wù)三個(gè)狀態(tài)之間直接切換時(shí)觸發(fā)此函數(shù)
}
@Override
public void onProviderEnabled(String provider) {
//Provider被enable時(shí)觸發(fā)此函數(shù),比如GPS被打開(kāi)
}
@Override
public void onProviderDisabled(String provider) {
//Provider被disable時(shí)觸發(fā)此函數(shù),比如GPS被關(guān)閉
}
}
}
因?yàn)橛玫搅说貓D服務(wù),所以需要在AndroidManifest.xml中的application標(biāo)簽之間加入google map library聲明:
<uses-library android:name="com.google.android.maps" />
然后加入位置服務(wù)所需的權(quán)限:
<uses-permission android:name="android.permission.ACCESS_FINE_LOCATION" />
這里朋友們需要注意:如果使用GPS_PROVIDER或者同時(shí)使用GPS_PROVIDER和NETWORK_PROVIDER,則只需聲明ACCESS_FINE_LOCATION權(quán)限,它對(duì)于上述兩個(gè)provider都是有效的;而ACCESS_COARSE_LOCATION權(quán)限只針對(duì)NETWORK_PROVIDER。
如果是在模擬器里調(diào)試的話,我們可以用以下兩種方法設(shè)置一個(gè)模擬的坐標(biāo)值:geo命令和DDMS。
先來(lái)說(shuō)一下geo命令,它需要telnet到本機(jī)的5554端口,然后在命令行下輸入geo fix命令,參數(shù)可附帶經(jīng)度、緯度和海拔(可選)。
具體操作如圖:


如果朋友用的系統(tǒng)是windows7的話,會(huì)遇到一些小小的麻煩,因?yàn)閣indows7默認(rèn)是沒(méi)有裝Telnet服務(wù),所以我們需要手動(dòng)安裝一下,點(diǎn)擊“開(kāi)始->控制面板->程序->程序和功能”,然后再?gòu)棾龅拇翱谧髠?cè)點(diǎn)擊“打開(kāi)或關(guān)閉Windows功能”,會(huì)彈出一下界面,選中“Telnet客戶端”和“Telnet服務(wù)端”即可。如圖:

不過(guò),使用geo命令還是挺麻煩的,ADT提供了一個(gè)設(shè)置模擬坐標(biāo)的界面,打開(kāi)“Emulator Control”視圖,即可看到一下界面:

如果設(shè)置了模擬坐標(biāo)后,在模擬器的狀態(tài)欄就會(huì)出現(xiàn)一個(gè)雷達(dá)圖形的標(biāo)志,如圖:

到此這篇關(guān)于Android Location服務(wù)之LocationManager案例詳解的文章就介紹到這了,更多相關(guān)Android Location服務(wù)之LocationManager內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
詳解Android的Splash啟動(dòng)圖的兩種動(dòng)態(tài)切換方式
本篇文章主要介紹了詳解Android的Splash啟動(dòng)圖的兩種動(dòng)態(tài)切換方式,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-06-06
Android?Activity生命周期調(diào)用的理解
大家好。本篇文章主要講的是Android?Activity生命周期調(diào)用的理解,感興趣的同學(xué)趕快來(lái)看一看吧,對(duì)你有幫助的話記得收藏一下,方便下次瀏覽2021-12-12
基于Flutter實(shí)現(xiàn)圖片選擇和圖片上傳
Flutter?的圖片選擇插件很多,包括了官方的?image_picker,multi_image_picker(基于2.0出了?multi_image_picker2)等等。本文將利用這些插件實(shí)現(xiàn)圖片選擇和圖片上傳,需要的可以參考一下2022-03-03
Android中g(shù)oogle Zxing實(shí)現(xiàn)二維碼與條形碼掃描
這篇文章主要介紹了Android中g(shù)oogle Zxing實(shí)現(xiàn)二維碼與條形碼掃描的相關(guān)資料,需要的朋友可以參考下2017-05-05
Android簡(jiǎn)單封裝一個(gè)MVP基類流程詳解
MVP是從經(jīng)典的模式MVC演變而來(lái),它們的基本思想有相通的地方:Controller/Presenter負(fù)責(zé)邏輯的處理,Model提供數(shù)據(jù),View負(fù)責(zé)顯示。下面這篇文章主要給大家介紹了關(guān)于Android從實(shí)現(xiàn)到封裝MVP的相關(guān)內(nèi)容,分享出來(lái)供大家參考學(xué)習(xí),下面話不多說(shuō)了,來(lái)一起看看詳細(xì)的介紹吧2023-03-03
Android開(kāi)發(fā)中判斷手機(jī)是否安裝了QQ或者微信
這篇文章主要介紹了Android開(kāi)發(fā)中判斷手機(jī)是否安裝了QQ或者微信的相關(guān)資料,需要的朋友可以參考下2017-01-01
Android巧用ActionBar實(shí)現(xiàn)tab導(dǎo)航效果
這篇文章主要為大家詳細(xì)介紹了Android巧用ActionBar實(shí)現(xiàn)tab導(dǎo)航效果,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2016-05-05

