我想构建一个以获取用户当前位置为中心的应用,然后通过 Google地方信息找到与他/她关系密切的兴趣点(如酒吧,餐馆等) API 即可。
在网上搜索要开始的地方后,我发现了一些使用LocationManager
课程的教程以及其他使用 Google Play服务的教程,以便找到用户所在的位置。< / p>
第一眼看到他们两个都做同样的事情,但由于我是新手,我有点困惑,我不知道哪种方法最适合我的需要。所以,我想问你:
这两种寻找位置的方法有什么区别(如果有的话)?
答案 0 :(得分:296)
Android上的用户位置
在Android上获取用户的位置比在iOS上简单明了一些。要开始混淆,有两种完全不同的方法可以做到。第一种是使用android.location.LocationListener
中的Android API,第二种是使用Google Play服务API com.google.android.gms.location.LocationListener
。让我们来看看他们两个。
Android的位置API
Android的位置API使用三种不同的提供商来获取位置 -
LocationManager.GPS_PROVIDER
- 此提供商使用卫星确定位置。根据条件,此提供商可能需要一段时间才能返回位置修复。LocationManager.NETWORK_PROVIDER
- 此提供商根据手机信号塔和WiFi接入点的可用性确定位置。通过网络查找检索结果。LocationManager.PASSIVE_PROVIDER
- 此提供程序将返回其他提供程序生成的位置。当其他应用程序或服务请求时,您会被动地接收位置更新,而无需自己实际请求这些位置。它的要点是,您从系统中获得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可以获得更好的电池性能以及更准确的准确度。现在混乱开始了!
Google的位置服务API是Google Play服务APK(here’s how to set it up)的一部分。它们建立在Android的API之上。这些API提供“融合位置提供程序”而不是上面提到的提供程序。此提供程序会根据准确性,电池使用情况等自动选择要使用的基础提供程序。它很快,因为您从系统范围的服务获取位置并不断更新它。您可以使用更高级的功能,例如地理围栏。
要使用Google的位置服务,您的应用需要连接到GooglePlayServicesClient
。要连接到客户端,您的活动(或片段等)需要实现GooglePlayServicesClient.ConnectionCallbacks
和GooglePlayServicesClient.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);
}
<强>更新强>
文档已更新,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没有提供关于个人职位如何得出的信息。
这使您不得不根据对速度,加速度,方位等的合理性检查过滤掉怪异值。
...或者回到旧的框架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
requestLocationUpdates()
方法来提取位置。locationRequest.setInterval(milliseconds)
和setFastestInterval(milliseconds)
,而不基于用户位置更改 LocationManager Api
根据用户位置更改和时间间隔 locationManager.requestLocationUpdates(LocationManager.GPS_PROVIDER, milliseconds, mindistance, Mylocationlistener)
返回的LatLng值包含 14个十进制值 (例如:11.94574594963342 79.81166719458997) 准确的位置值
< / li>答案 7 :(得分:0)
LocationManager - Context.getSystemService(Context.LOCATION_SERVICE)
ACCESS_COARSE_LOCATION - 提供不太准确的位置(城市街区)但速度更快并且不会消耗太多电池
ACCESS_FINE_LOCATION - 提供更好、更准确的位置(最多几米)。使用与 ACCESS_COARSE_LOCATION 相同的提供程序
Google API 位置服务 - GoogleApiClient
基于 Google Play 服务。可以访问通过系统的位置事件的高级 API。它具有更好的电池性能,但无法安装在某些设备上