9.3.2 LocationManager应用示例

本节所使用的示例运行后的界面如图9-30所示。

9.3.2 LocationManager应用示例 - 图1

图9-30 示例运行效果

·左边按钮为"Fine-grain provider",表示通过GpsLP来获取位置信息。

·右边按钮为"Both Providers",表示同时使用GpsLP和NetworkLP来获取位置信息。

·按钮下方的"Lat/Long"表示当前设备的位置信息(经纬度值),而"Address"表示根据该位置信息得到的地址信息。

示例非常简单,所有内容都集中在LocationActivity.java文件中。先来看onCreate函数,代码如下所示。

[—>LocationActivity.java::onCreate]

  1. public void onCreate(Bundle savedInstanceState) {
  2. super.onCreate(savedInstanceState);
  3. setContentView(R.layout.main);
  4. ......// UI 等初始化
  5. // Geocoder:只有Android 2.3版本以后系统才支持该功能
  6. // 同时还需要判断是否存在GeocoderProvider
  7. mGeocoderAvailable =
  8. Build.VERSION.SDK_INT >= Build.VERSION_CODES.GINGERBREAD &&
  9. Geocoder.isPresent();
  10.  
  11. mHandler = new Handler() {......// Handler的作用是更新图9-30中的位置和地址信息};
  12. // 客户端必须要获取LocationManager来和LMS交互
  13. mLocationManager = (LocationManager) getSystemService(Context.LOCATION_SERVICE);
  14. }

接着,LocationActivity在其onResume中将调用setup函数完成进一步的初始化工作,其代码如下所示。

[—>LocationActivity.java::setup]

  1. private void setup() {
  2. Location gpsLocation = null; Location networkLocation = null;
  3. mLocationManager.removeUpdates(listener);
  4. mLatLng.setText(R.string.unknown); mAddress.setText(R.string.unknown);
  5. if (mUseFine) { // mUseFine对应为图9-30中的“Fine-grain Provider”按钮
  6. ......
  7. // requestUpdatesFromProvider为关键函数,下文将详细分析
  8. gpsLocation = requestUpdatesFromProvider(
  9. LocationManager.GPS_PROVIDER,// 该参数为字符串,值为“gps”
  10. R.string.not_support_gps);
  11. if (gpsLocation != null) updateUILocation(gpsLocation);
  12. } else if (mUseBoth) {// mUseBoth对应为图9-30中的“Both Providers”按钮
  13. gpsLocation = requestUpdatesFromProvider(
  14. LocationManager.GPS_PROVIDER, R.string.not_support_gps);
  15. networkLocation = requestUpdatesFromProvider(
  16. LocationManager.NETWORK_PROVIDER, // 该参数值为“network”
  17. R.string.not_support_network);
  18. ......
  19.  
  20. }
  21. }

看setup中的关键函数requestUpdatesFromProvider代码如下所示。

[—>LocationActivity.java::requestUpdatesFromProvider]

  1. private Location requestUpdatesFromProvider(final String provider,
  2. final int errorResId) {
  3. Location location = null;
  4. // 判断由provider指定的LP是否启用。
  5. if (mLocationManager.isProviderEnabled(provider)) {
  6. /*
  7. 调用LocationManager的requestLocationUpdates函数,该函数用于注册一个回调函数以
  8. 接收位置变化信息,其各个参数的作用如下。
  9. provider:用于指明使用哪个LP,目前可取参数有“gps”(对应为GpsLP)、“network”(对应
  10. 为NetworkLP)、“passive”(对应为PassiveProvider)。
  11. TEN_SECONDS:用于指明多少毫秒更新一次位置数据,本例中使用的值为10秒。
  12. TEN_METERS:用于指明位置变化多少时更新一次数据,本例中使用的值为10米。
  13. listener:类型为LocationListener,当位置发生变化时,其onLocationChanged函数将被调用。
  14. */
  15. mLocationManager.requestLocationUpdates(provider, TEN_SECONDS,
  16. TEN_METERS,listener);
  17. // getLastKnownLocation用于获取LP上一次保存的位置信息数据
  18. // Android平台中,位置信息用Location类表示
  19. location = mLocationManager.getLastKnownLocation(provider);
  20. }
  21. return location;
  22. }

requestLocationUpdates是LM中非常重要的函数,请读者务必把握其用法。

当位置信息发生变化后,LocationListener的onChange函数将被调用。本例中使用的LocationListener相关代码如下所示。

[—>LocationActivity.java::LocationListener]

  1. private final LocationListener listener = new LocationListener() {
  2. public void onLocationChanged(Location location) {
  3. updateUILocation(location);// 下文将分析它
  4. }
  5. // 当用户在设置中启用或禁止相关LocationProvider后,下面这两个函数将被调用
  6. public void onProviderEnabled(String provider) { }
  7. public void onProviderDisabled(String provider) { }
  8. // 当LocationProvider的状态发生变化时,下面这个函数将被调用
  9. public void onStatusChanged(String provider, int status, Bundle extras) {}
  10. };

updateUILocation函数代码如下所示。

[—>LocationActivity.java::updateUILocation]

  1. private void updateUILocation(Location location) {
  2. // updateUILocation的参数location代表对应LP得到的位置信息
  3. // 示例程序将根据该信息来更新图9-30中“Lat/Long”的值
  4. Message.obtain(mHandler, UPDATE_LATLNG,
  5. location.getLatitude() + ", " + location.getLongitude()).sendToTarget();
  6.  
  7. // doReverseGeocoding根据位置信息来获取地址信息
  8. if (mGeocoderAvailable) doReverseGeocoding(location);
  9. }

doReverseGeocoding用于根据位置信息来获取地址信息,由于该工作往往需要通过网络来查询,所以doReverseGeocoding内部将创建一个AsyncTask用来完成此工作。我们直接来看AsyncTask的代码。

[—>LocationActivity.java::ReverseGeocodingTask]

  1. private class ReverseGeocodingTask extends AsyncTask<Location, Void, Void> {
  2. Context mContext;
  3. ......
  4. protected Void doInBackground(Location... params) {
  5. // 创建一个Geocoder对象,它可用于处理地址信息和位置信息的转换
  6. Geocoder geocoder = new Geocoder(mContext, Locale.getDefault());
  7. Location loc = params[0];
  8. List<Address> addresses = null;
  9. try {
  10. // getFromLocation用于根据位置信息来查询对应的地址信息
  11. addresses = geocoder.getFromLocation(loc.getLatitude(), loc.getLongitude(), 1);
  12. } ......
  13. if (addresses != null && addresses.size() > 0) {
  14. Address address = addresses.get(0);
  15. String addressText = String.format("%s, %s, %s",
  16. address.getMaxAddressLineIndex() > 0 ? address.getAddressLine(0) : "",
  17. address.getLocality(),address.getCountryName());
  18. // 更新图9-30中的“Address”信息
  19. Message.obtain(mHandler, UPDATE_ADDRESS, addressText).sendToTarget();
  20. }
  21. return null;
  22. }
  23. }

通过上述示例可以发现,Android平台中使用LM非常简单,其主要工作如下。

1)先创建一个LocationManager对象,用于和LMS交互。

2)然后调用requestLocationUpdates以设置一个回调接口对象LocationListener,同时还需要指明使用哪个LP

3)当LP更新相关信息后,LocationListener对应的函数将被调用,应用程序在这些回调函数中做相关处理即可。

4)如果应用程序需要在位置和地址信息做转换,则使用Geocoder类提供的函数即可。

虽然LM比较简单,但它提供的都是一些很基本的功能,如果想实现一些诸如显示地图信息这样的功能,LM就无能为力了。为了实现一些更复杂的位置相关的功能,Google提供了更高级的API来帮助开发者。关于这一部分内容,建议读者阅读参考资料[30]。

 根据审稿专家的意见,有些应用会只传进定位的条件(精度),而不去指定使用哪个LP。在不带GPS功能的平台,若应用只指定从GPS中获取定位数据,则它将得不到位置信息。