Android:LocationManager与Google Play服务

时间:2015-10-08 17:48:08

标签: android gps geolocation

我想构建一个以获取用户当前位置为中心的应用,然后通过 Google地方信息找到与他/她关系密切的兴趣点(如酒吧,餐馆等) API 即可。

在网上搜索要开始的地方后,我发现了一些使用LocationManager课程的教程以及其他使用 Google Play服务的教程,以便找到用户所在的位置。< / p>

第一眼看到他们两个都做同样的事情,但由于我是新手,我有点困惑,我不知道哪种方法最适合我的需要。所以,我想问你:

这两种寻找位置的方法有什么区别(如果有的话)?

8 个答案:

答案 0 :(得分:296)

Android上的用户位置

在Android上获取用户的位置比在iOS上简单明了一些。要开始混淆,有两种完全不同的方法可以做到。第一种是使用android.location.LocationListener中的Android API,第二种是使用Google Play服务API com.google.android.gms.location.LocationListener。让我们来看看他们两个。

  1. Android的位置API

    Android的位置API使用三种不同的提供商来获取位置 -

    • LocationManager.GPS_PROVIDER - 此提供商使用卫星确定位置。根据条件,此提供商可能需要一段时间才能返回位置修复。
    • LocationManager.NETWORK_PROVIDER - 此提供商根据手机信号塔和WiFi接入点的可用性确定位置。通过网络查找检索结果。
    • LocationManager.PASSIVE_PROVIDER - 此提供程序将返回其他提供程序生成的位置。当其他应用程序或服务请求时,您会被动地接收位置更新,而无需自己实际请求这些位置。
  2. 它的要点是,您从系统中获得LocationManager的对象,实施LocationListener,并在requestLocationUpdates上调用LocationManager

    以下是代码段:

        LocationManager locationManager = (LocationManager) this.getSystemService(Context.LOCATION_SERVICE);
    // Define a listener that responds to location updates
    LocationListener locationListener = new LocationListener() {
        public void onLocationChanged(Location location) {
          // Called when a new location is found by the network location provider.
          makeUseOfNewLocation(location);
        }
    
        public void onStatusChanged(String provider, int status, Bundle extras) {}
    
        public void onProviderEnabled(String provider) {}
    
        public void onProviderDisabled(String provider) {}
      };
    
    // Register the listener with the Location Manager to receive location updates
    locationManager.requestLocationUpdates(LocationManager.NETWORK_PROVIDER, 0, 0, locationListener);
    

    Google’s API Guide on Location Strategies很好地解释了代码。但他们也提到,在大多数情况下,使用Google Location Services API可以获得更好的电池性能以及更准确的准确度。现在混乱开始了!

    1. Google的位置服务API
    2. Google的位置服务API是Google Play服务APK(here’s how to set it up)的一部分。它们建立在Android的API之上。这些API提供“融合位置提供程序”而不是上面提到的提供程序。此提供程序会根据准确性,电池使用情况等自动选择要使用的基础提供程序。它很快,因为您从系统范围的服务获取位置并不断更新它。您可以使用更高级的功能,例如地理围栏。

      要使用Google的位置服务,您的应用需要连接到GooglePlayServicesClient。要连接到客户端,您的活动(或片段等)需要实现GooglePlayServicesClient.ConnectionCallbacksGooglePlayServicesClient.OnConnectionFailedListener接口。 这是一个示例代码:

          public class MyActivity extends Activity implements ConnectionCallbacks, OnConnectionFailedListener {
          LocationClient locationClient;
      
          @Override
          protected void onCreate(Bundle savedInstanceState) {
              super.onCreate(savedInstanceState);
              setContentView(R.layout.activity_my);
              locationClient = new LocationClient(this, this, this);
          }
      
          @Override
          public void onConnected(Bundle bundle) {
          Location location = locationClient.getLastLocation() ;
              Toast.makeText(this, "Connected to Google Play Services", Toast.LENGTH_SHORT).show();
          }
      
          @Override
          public void onDisconnected() {
          Toast.makeText(this, "Connected from Google Play Services.", Toast.LENGTH_SHORT).show();
          }
      
          @Override
          public void onConnectionFailed(ConnectionResult connectionResult) {
              // code to handle failed connection
              // this code can be found here — http://developer.android.com/training/location/retrieve-current.html 
          }
      
      • 为什么locationClient.getLastLocation()为空?

      locationClient.getLastLocation()从客户端获取最后一个已知位置。但是,如果至少有一个客户端连接到它,则融合位置提供程序将仅保留后台位置。一旦第一个客户端连接,它将立即尝试获取位置。如果您的活动是第一个连接的客户,并且您在getLastLocation()中立即致电onConnected(),则可能没有足够的时间让第一个地点进入。这将导致location成为null

      要解决此问题,您必须等待(不确定),直到提供者获取位置,然后调用getLastLocation(),这是不可能知道的。另一个(更好)选项是实现com.google.android.gms.location.LocationListener接口以接收定期位置更新(并在第一次更新后将其关闭)。

          public class MyActivity extends Activity implements ConnectionCallbacks, OnConnectionFailedListener, LocationListener {
          // . . . . . . . . more stuff here 
          LocationRequest locationRequest;
          LocationClient locationClient;
      
          @Override
          protected void onCreate(Bundle savedInstanceState) {
              // . . . . other initialization code
              locationClient = new LocationClient(this, this, this);
          locationRequest = new LocationRequest();
          // Use high accuracy
          locationRequest.setPriority(LocationRequest.PRIORITY_HIGH_ACCURACY);
              // Set the update interval to 5 seconds
          locationRequest.setInterval(UPDATE_INTERVAL);
              // Set the fastest update interval to 1 second
          locationRequest.setFastestInterval(FASTEST_INTERVAL);
          }
          // . . . . . . . . other methods 
          @Override
          public void onConnected(Bundle bundle) {
              Location location = locationClient.getLastLocation();
              if (location == null)
                  locationClient.requestLocationUpdates(locationRequest, this);
              else
                  Toast.makeText(getActivity(), "Location: " + location.getLatitude() + ", " + location.getLongitude(), Toast.LENGTH_SHORT).show();
          }
          // . . . . . . . . other methods
          @Override
          public void onLocationChanged(Location location) {
              locationClient.removeLocationUpdates(this);
              // Use the location here!!!
          }
      

      在此代码中,您要检查客户端是否已有最后一个位置(在onConnected中)。如果没有,您需要进行位置更新,并在获得更新后立即关闭请求(在onLocationChanged()回调中)。

      请注意,locationClient.requestLocationUpdates(locationRequest, this);必须位于onConnected回调中,否则您将获得IllegalStateException,因为您将尝试申请未连接到Google Play服务的位置客户端。

      • 用户已停用位置服务

      很多时候,用户会禁用位置服务(以节省电池或隐私原因)。在这种情况下,上面的代码仍会请求位置更新,但永远不会调用onLocationChanged。您可以通过检查用户是否已禁用位置服务来停止请求。

      如果您的应用要求他们启用位置服务,您可能希望显示消息或祝酒。很遗憾,无法检查用户是否在Google的位置服务API中禁用了位置服务。为此,您将不得不使用Android的API。

      onCreate方法中:

          LocationManager manager = (LocationManager) getActivity().getSystemService(Context.LOCATION_SERVICE);
      if (!manager.isProviderEnabled(LocationManager.GPS_PROVIDER) && !manager.isProviderEnabled(LocationManager.NETWORK_PROVIDER)) {
          locationEnabled = false;
          Toast.makeText(getActivity(), "Enable location services for accurate data", Toast.LENGTH_SHORT).show();
      }
      else locationEnabled = true;
      

      并在locationEnabled方法中使用onConnected标记,如下所示:

          if (location != null) {
          Toast.makeText(getActivity(), "Location: " + location.getLatitude() + ", " + location.getLongitude(), Toast.LENGTH_SHORT).show();
      }
      else if (location == null && locationEnabled) {
          locationClient.requestLocationUpdates(locationRequest, this);
      }
      

      感谢Rahul Jiresal

      <强>更新

      文档已更新,LocationClient已删除,api支持通过对话框单击启用GPS:

      task.addOnSuccessListener(this, new OnSuccessListener<LocationSettingsResponse>() {
      @Override
      public void onSuccess(LocationSettingsResponse locationSettingsResponse) {
          // All location settings are satisfied. The client can initialize
          // location requests here.
          // ...
      }
      });
      
      task.addOnFailureListener(this, new OnFailureListener() {
          @Override
          public void onFailure(@NonNull Exception e) {
              if (e instanceof ResolvableApiException) {
                  // Location settings are not satisfied, but this can be fixed
                  // by showing the user a dialog.
                  try {
                      // Show the dialog by calling startResolutionForResult(),
                      // and check the result in onActivityResult().
                      ResolvableApiException resolvable = (ResolvableApiException) e;
                      resolvable.startResolutionForResult(MainActivity.this,
                              REQUEST_CHECK_SETTINGS);
                  } catch (IntentSender.SendIntentException sendEx) {
                      // Ignore the error.
                  }
              }
          }
      });
      

      链接https://developer.android.com/training/location/change-location-settings#prompt

      新位置客户端:FusedLocationProviderClient

        private FusedLocationProviderClient fusedLocationClient;
      
      @Override
      protected void onCreate(Bundle savedInstanceState) {
          fusedLocationClient = LocationServices.getFusedLocationProviderClient(this);
      }
      

      建议在执行任何位置任务之前通过https://developer.android.com/training/location

答案 1 :(得分:27)

根据我的经验,“更合适的准确性”并不意味着更好。除非我遗漏了什么,否则如果你想确保使用GPS,那么LocationManager是唯一的方法。我们使用我们的应用程序跟踪车辆,除非我遗漏了某些内容,否则Google Play服务会经常提供一些非常不准确的位置。

答案 2 :(得分:25)

您应该使用Google Play服务位置API而不是LocationManager。根据文件:

  

Google Play服务位置API优于Android   框架位置API(android.location)作为添加位置的一种方式   了解您的应用。如果您目前正在使用Android   框架位置API,强烈建议您切换到   Google Play尽快提供服务位置API。

至于切换的原因,谷歌说:

  

Google位置服务API,Google Play服务的一部分,   提供更强大的自动高级框架   处理位置提供者,用户移动和位置准确性。它   还根据功耗处理位置更新调度   您提供的参数。在大多数情况下,你会获得更好的电池   通过使用,性能,以及更合适的准确性   位置服务API。

答案 3 :(得分:17)

我一直在使用Google位置服务API。它确实具有优势,因为它封装了具有用于确定位置的若干源的复杂性。但是封装太多,所以当你得到一个奇怪的位置时,你无法确定那个奇怪的位置来自哪里。

在现实生活中,我有几个奇怪的值弹出,距实际位置10公里。唯一的解释是,这些疯狂的位置源于谷歌Wi-Fi或NWK数据库中的错误 - 由于Wi-Fi和网络拓扑每天都在变化,因此总会出现错误。但遗憾的是(并且令人惊讶地)API没有提供关于个人职位如何得出的信息。

enter image description here

这使您不得不根据对速度,加速度,方位等的合理性检查过滤掉怪异值。

...或者回到旧的框架API并仅使用GPS,这是我决定在Google改进融合API之前做的事情。

答案 4 :(得分:6)

Google位置服务API ,是Google Play服务的一部分,提供更强大的高级框架,可自动处理位置提供商用户移动位置准确性。它还根据您提供的功耗参数处理位置更新计划。在大多数情况下,使用位置服务API,您将获得 更好的电池性能 ,以及更合适的准确度。

可以找到两个apis Google Play服务位置API Android Framework位置API 之间更详细的差异here

答案 5 :(得分:0)

是的,Google Play服务位置服务API可能会误导位置信息。 WiFi调制解调器被移动,WiFi调制解调器被更新了不正确的位置信息(即,如果位置被更新WiFi调制解调器的位置的Android设备所欺骗),并且在许多其他情况下,可能会导致WiFi调制解调器三角剖分产生不正确的位置数据。在我们所有要求精确定位的应用中,我们仅使用GPS。

答案 6 :(得分:0)

基于 GPS服务

的两个API Google Play服务位置API和Android框架位置API之间的区别

FusedLocationProviderClient

  1. 对于第一次抓取,该位置不能为空(即:某些其他应用需要更新GoogleplayService数据库中的Lastknown位置。如果为空,则需要变通)
  2. 对于下一次顺序提取,它使用requestLocationUpdates()方法来提取位置。
  3. 位置提取仅基于locationRequest.setInterval(milliseconds)setFastestInterval(milliseconds),而不基于用户位置更改
  4. 返回的LatLng值仅包含 7个十进制值(例如:11.9557996、79.8234599),不准确
  5. 推荐,当您的应用程序要求使用当前位置时可以忽略的距离(精度为50-100米)
  6. 有效使用电池。

LocationManager Api

  1. 使用locationManager.requestLocationUpdates()调用用户位置提取
  2. 根据用户位置更改时间间隔 locationManager.requestLocationUpdates(LocationManager.GPS_PROVIDER, milliseconds, mindistance, Mylocationlistener)

  3. 获取位置
  4. 返回的LatLng值包含 14个十进制值 (例如:11.94574594963342 79.81166719458997) 准确的位置值

    < / li>
  5. 推荐,用于基于位置的应用,即使在以米为单位的情况下也需要更高的准确性。
  6. 电池使用情况取决于获取间隔和获取距离。

答案 7 :(得分:0)

LocationManager - Context.getSystemService(Context.LOCATION_SERVICE)

  • ACCESS_COARSE_LOCATION - 提供不太准确的位置(城市街区)但速度更快并且不会消耗太多电池

    • NETWORK_PROVIDER - 使用手机信号塔、wifi 接入点
    • PASSIVE_PROVIDER - 当系统中的其他人使用其他提供者时订阅位置更新
  • ACCESS_FINE_LOCATION - 提供更好、更准确的位置(最多几米)。使用与 ACCESS_COARSE_LOCATION 相同的提供程序

    • GPS_PROVIDER - 使用卫星

Google API 位置服务 - GoogleApiClient 基于 Google Play 服务。可以访问通过系统的位置事件的高级 API。它具有更好的电池性能,但无法安装在某些设备上

  • 融合位置提供商 - 根据您的需求和设备条件自动选择合适的提供商