本節所使用的示例運行后的界面如圖9-30所示。
:-: 
圖9-30 示例運行效果
* 左邊按鈕為"Fine-grain provider",表示通過GpsLP來獲取位置信息。
* 右邊按鈕為"Both Providers",表示同時使用GpsLP和NetworkLP來獲取位置信息。
* 按鈕下方的"Lat/Long"表示當前設備的位置信息(經緯度值),而"Address"表示根據該位置信息得到的地址信息。
示例非常簡單,所有內容都集中在LocationActivity.java文件中。先來看onCreate函數,代碼如下所示。
**LocationActivity.java::onCreate**
~~~
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.main);
......// UI 等初始化
// Geocoder:只有Android 2.3版本以后系統才支持該功能
// 同時還需要判斷是否存在GeocoderProvider
mGeocoderAvailable =
Build.VERSION.SDK_INT >= Build.VERSION_CODES.GINGERBREAD &&
Geocoder.isPresent();
mHandler = new Handler() {......// Handler的作用是更新圖9-30中的位置和地址信息};
// 客戶端必須要獲取LocationManager來和LMS交互
mLocationManager = (LocationManager) getSystemService(Context.LOCATION_SERVICE);
}
}
~~~
接著,LocationActivity在其onResume中將調用setup函數完成進一步的初始化工作,其代碼如下所示。
**LocationActivity.java::setup**
~~~
private void setup() {
Location gpsLocation = null; Location networkLocation = null;
mLocationManager.removeUpdates(listener);
mLatLng.setText(R.string.unknown); mAddress.setText(R.string.unknown);
if (mUseFine) { // mUseFine對應為圖9-30中的“Fine-grain Provider”按鈕
......
// requestUpdatesFromProvider為關鍵函數,下文將詳細分析
gpsLocation = requestUpdatesFromProvider(
LocationManager.GPS_PROVIDER,// 該參數為字符串,值為“gps”
R.string.not_support_gps);
if (gpsLocation != null) updateUILocation(gpsLocation);
} else if (mUseBoth) {// mUseBoth對應為圖9-30中的“Both Providers”按鈕
gpsLocation = requestUpdatesFromProvider(LocationManager.GPS_PROVIDER, R.string.not_support_gps);
networkLocation = requestUpdatesFromProvider(
LocationManager.NETWORK_PROVIDER, // 該參數值為“network”
R.string.not_support_network);
......
}
}
~~~
看setup中的關鍵函數requestUpdatesFromProvider代碼如下所示。
**LocationActivity.java::requestUpdatesFromProvider**
~~~
private Location requestUpdatesFromProvider(final String provider,
final int errorResId) {
Location location = null;
// 判斷由provider指定的LP是否啟用。
if (mLocationManager.isProviderEnabled(provider)) {
/*
調用LocationManager的requestLocationUpdates函數,該函數用于注冊一個回調函數以
接收位置變化信息,其各個參數的作用如下。
provider:用于指明使用哪個LP,目前可取參數有“gps”(對應為GpsLP)、“network”(對應
為NetworkLP)、“passive”(對應為PassiveProvider)。
TEN_SECONDS:用于指明多少毫秒更新一次位置數據,本例中使用的值為10秒。
TEN_METERS:用于指明位置變化多少時更新一次數據,本例中使用的值為10米。
listener:類型為LocationListener,當位置發生變化時,其onLocationChanged函數將被調用。
*/
mLocationManager.requestLocationUpdates(provider, TEN_SECONDS,
TEN_METERS,listener);
// getLastKnownLocation用于獲取LP上一次保存的位置信息數據
// Android平臺中,位置信息用Location類表示
location = mLocationManager.getLastKnownLocation(provider);
}
return location;
}
~~~
requestLocationUpdates是LM中非常重要的函數,請讀者務必把握其用法。
當位置信息發生變化后,LocationListener的onChange函數將被調用。本例中使用的LocationListener相關代碼如下所示。
**LocationActivity.java::LocationListener**
~~~
private final LocationListener listener = new LocationListener() {
public void onLocationChanged(Location location) {
updateUILocation(location);// 下文將分析它
}
// 當用戶在設置中啟用或禁止相關LocationProvider后,下面這兩個函數將被調用
public void onProviderEnabled(String provider) { }
public void onProviderDisabled(String provider) { }
// 當LocationProvider的狀態發生變化時,下面這個函數將被調用
public void onStatusChanged(String provider, int status, Bundle extras) {}
};
~~~
updateUILocation函數代碼如下所示。
**LocationActivity.java::updateUILocation**
~~~
private void updateUILocation(Location location) {
// updateUILocation的參數location代表對應LP得到的位置信息
// 示例程序將根據該信息來更新圖9-30中“Lat/Long”的值
Message.obtain(mHandler, UPDATE_LATLNG,
location.getLatitude() + ", " + location.getLongitude()).sendToTarget();
// doReverseGeocoding根據位置信息來獲取地址信息
if (mGeocoderAvailable) doReverseGeocoding(location);
}
~~~
doReverseGeocoding用于根據位置信息來獲取地址信息,由于該工作往往需要通過網絡來查詢,所以doReverseGeocoding內部將創建一個AsyncTask用來完成此工作。我們直接來看AsyncTask的代碼。
**LocationActivity.java::ReverseGeocodingTask**
~~~
private class ReverseGeocodingTask extends AsyncTask<Location, Void, Void> {
Context mContext;
......
protected Void doInBackground(Location... params) {
// 創建一個Geocoder對象,它可用于處理地址信息和位置信息的轉換
Geocoder geocoder = new Geocoder(mContext, Locale.getDefault());
Location loc = params[0];
List<Address> addresses = null;
try {
// getFromLocation用于根據位置信息來查詢對應的地址信息
addresses = geocoder.getFromLocation(loc.getLatitude(), loc.getLongitude(), 1);
} ......
if (addresses != null && addresses.size() > 0) {
Address address = addresses.get(0);
String addressText = String.format("%s, %s, %s",
address.getMaxAddressLineIndex() > 0 ? address.getAddressLine(0) : "",
address.getLocality(),address.getCountryName());
// 更新圖9-30中的“Address”信息
Message.obtain(mHandler, UPDATE_ADDRESS, addressText).sendToTarget();
}
return null;
}
}
~~~
通過上述示例可以發現,Android平臺中使用LM非常簡單,其主要工作如下。
1. 先創建一個LocationManager對象,用于和LMS交互。
2. 然后調用requestLocationUpdates以設置一個回調接口對象LocationListener,同時還需要指明使用哪個LP[^①]。
3. 當LP更新相關信息后,LocationListener對應的函數將被調用,應用程序在這些回調函數中做相關處理即可。
4. 如果應用程序需要在位置和地址信息做轉換,則使用Geocoder類提供的函數即可。
雖然LM比較簡單,但它提供的都是一些很基本的功能,如果想實現一些諸如顯示地圖信息這樣的功能,LM就無能為力了。為了實現一些更復雜的位置相關的功能,Google提供了更高級的API來幫助開發者。關于這一部分內容,建議讀者閱讀參考資料[30]。
[^①]:根據審稿專家的意見,有些應用會只傳進定位的條件(精度),而不去指定使用哪個LP。在不帶GPS功能的平臺,若應用只指定從GPS中獲取定位數據,則它將得不到位置信息。
- 前言
- 第1章 準備工作
- 1.1 Android系統架構
- 1.2 工具使用
- 1.2.1 Source Insight的使用
- 1.2.2 Eclipse的使用
- 1.2.3 BusyBox的使用
- 1.3 本書資源下載說明
- 第2章 深入理解Netd
- 2.1 概述
- 2.2 Netd工作流程
- 2.2.1 main函數分析
- 2.2.2 NetlinkManager分析
- 2.2.3 CommandListener分析
- 2.2.4 DnsProxyListener分析
- 2.2.5 MDnsSdListener分析
- 2.3 CommandListener中的命令
- 2.3.1 iptables、tc和ip命令
- 2.3.2 CommandListener構造函數和測試工具ndc
- 2.3.3 InterfaceCmd命令
- 2.3.4 IpFwd和FirewallCmd命令
- 2.3.5 ListTtysCmd和PppdCmd命令
- 2.3.6 BandwidthControlCmd和IdletimerControlCmd命令
- 2.3.7 NatCmd命令
- 2.3.8 TetherCmd和SoftapCmd命令
- 2.3.9 ResolverCmd命令
- 2.4 NetworkManagementService介紹
- 2.4.1 create函數詳解
- 2.4.2 systemReady函數詳解
- 2.5 本章總結和參考資料說明
- 2.5.1 本章總結
- 2.5.2 參考資料說明
- 第3章 Wi-Fi基礎知識
- 3.1 概述
- 3.2 無線電頻譜和802.11協議的發展歷程
- 3.2.1 無線電頻譜知識
- 3.2.2 IEEE 802.11發展歷程
- 3.3 802.11無線網絡技術
- 3.3.1 OSI基本參考模型及相關基本概念
- 3.3.2 802.11知識點導讀
- 3.3.3 802.11組件
- 3.3.4 802.11 Service介紹
- 3.3.5 802.11 MAC服務和幀
- 3.3.6 802.11 MAC管理實體
- 3.3.7 無線網絡安全技術知識點
- 3.4 Linux Wi-Fi編程API介紹
- 3.4.1 Linux Wireless Extensions介紹
- 3.4.2 nl80211介紹
- 3.5 本章總結和參考資料說明
- 3.5.1 本章總結
- 3.5.2 參考資料說明
- 第4章 深入理解wpa_supplicant
- 4.1 概述
- 4.2 初識wpa_supplicant
- 4.2.1 wpa_supplicant架構
- 4.2.2 wpa_supplicant編譯配置
- 4.2.3 wpa_supplicant命令和控制API
- 4.2.4 git的使用
- 4.3 wpa_supplicant初始化流程
- 4.3.1 main函數分析
- 4.3.2 wpa_supplicant_init函數分析
- 4.3.3 wpa_supplicant_add_iface函數分析
- 4.3.4 wpa_supplicant_init_iface函數分析
- 4.4 EAP和EAPOL模塊
- 4.4.1 EAP模塊分析
- 4.4.2 EAPOL模塊分析
- 4.5 wpa_supplicant連接無線網絡分析
- 4.5.1 ADD_NETWORK命令處理
- 4.5.2 SET_NETWORK命令處理
- 4.5.3 ENABLE_NETWORK命令處理
- 4.6 本章總結和參考資料說明
- 4.6.1 本章總結
- 4.6.2 參考資料說明
- 第5章 深入理解WifiService
- 5.1 概述
- 5.2 WifiService的創建及初始化
- 5.2.1 HSM和AsyncChannel介紹
- 5.2.2 WifiService構造函數分析
- 5.2.3 WifiStateMachine介紹
- 5.3 加入無線網絡分析
- 5.3.1 Settings操作Wi-Fi分析
- 5.3.2 WifiService操作Wi-Fi分析
- 5.4 WifiWatchdogStateMachine介紹
- 5.5 Captive Portal Check介紹
- 5.6 本章總結和參考資料說明
- 5.6.1 本章總結
- 5.6.2 參考資料說明
- 第6章 深入理解Wi-Fi Simple Configuration
- 6.1 概述
- 6.2 WSC基礎知識
- 6.2.1 WSC應用場景
- 6.2.2 WSC核心組件及接口
- 6.3 Registration Protocol詳解
- 6.3.1 WSC IE和Attribute介紹
- 6.3.2 802.11管理幀WSC IE設置
- 6.3.3 EAP-WSC介紹
- 6.4 WSC代碼分析
- 6.4.1 Settings中的WSC處理
- 6.4.2 WifiStateMachine的處理
- 6.4.3 wpa_supplicant中的WSC處理
- 6.4.4 EAP-WSC處理流程分析
- 6.5 本章總結和參考資料說明
- 6.5.1 本章總結
- 6.5.2 參考資料說明
- 第7章 深入理解Wi-Fi P2P
- 7.1 概述
- 7.2 P2P基礎知識
- 7.2.1 P2P架構
- 7.2.2 P2P Discovery技術
- 7.2.3 P2P工作流程
- 7.3 WifiP2pSettings和WifiP2pService介紹
- 7.3.1 WifiP2pSettings工作流程
- 7.3.2 WifiP2pService工作流程
- 7.4 wpa_supplicant中的P2P
- 7.4.1 P2P模塊初始化
- 7.4.2 P2P Device Discovery流程分析
- 7.4.3 Provision Discovery流程分析
- 7.4.4 GO Negotiation流程分析
- 7.5 本章總結和參考資料說明
- 7.5.1 本章總結
- 7.5.2 參考資料說明
- 第8章 深入理解NFC
- 8.1 概述
- 8.2 NFC基礎知識
- 8.2.1 NFC概述
- 8.2.2 NFC R/W運行模式
- 8.2.3 NFC P2P運行模式
- 8.2.4 NFC CE運行模式
- 8.2.5 NCI原理
- 8.2.6 NFC相關規范
- 8.3 Android中的NFC
- 8.3.1 NFC應用示例
- 8.3.2 NFC系統模塊
- 8.4 NFC HAL層討論
- 8.5 本章總結和參考資料說明
- 8.5.1 本章總結
- 8.5.2 參考資料說明
- 第9章 深入理解GPS
- 9.1 概述
- 9.2 GPS基礎知識
- 9.2.1 衛星導航基本原理
- 9.2.2 GPS系統組成及原理
- 9.2.3 OMA-SUPL協議
- 9.3 Android中的位置管理
- 9.3.1 LocationManager架構
- 9.3.2 LocationManager應用示例
- 9.3.3 LocationManager系統模塊
- 9.4 本章總結和參考資料說明
- 9.4.1 本章總結
- 9.4.2 參考資料說明
- 附錄