- Java 双重比较
- java - 比较器与 Apache BeanComparator
- Objective-C 完成 block 导致额外的方法调用?
- database - RESTful URI 是否应该公开数据库主键?
我有一个地理围栏应用程序,我正尝试将其移植到 Android 8+ 上。我读过 tutorial关于这个主题,并切换到使用 compile 'com.google.android.gms:play-services-location:16.0.0'
。
当应用程序不在前台时,地理围栏进入事件永远不会触发。我等待的时间超过了文档所说的两分钟。我已经等了 15 分钟,但没有结果。只要我将应用程序带到地理围栏内的前台,它就会立即触发。
我知道在 Android 8+ 上对后台服务有限制,但谷歌的教程说要使用在 Android 8+ 上应该被阻止的 IntentService。 后见之明编辑:上述教程绝对错误。不要跟随它。 IntentService 将不起作用。
我尝试关注 this answer ,它说使用新的 GeofencingClient
来设置您的 Intent 来解决这个问题。 (我不明白为什么这会有帮助,因为它仍然使用 IntentService 来接收事件,并且它在我的代码中不起作用。)
这个对相关问题的回答 here建议您必须使用 BroadcastReceiver 而不是 IntentService,但我不明白为什么这会有所不同,因为 Android 8 在后台对 BroadcastReceivers 施加了与 IntentServices 相同的限制。 事后编辑:这个链接的答案也是正确的。虽然隐式 BroadcastReceivers 受到限制,但在运行时注册以向特定包发送请求的显式 BroadcastReceivers 不受限制并且可以正常工作。
是否真的有可能在 Android 8+ 上获得在后台唤醒您的应用程序的地理围栏回调?如果是这样,您如何才能做到这一点?
我的地理围栏设置:
googleApiClient = new GoogleApiClient.Builder(this.context)
.addApi(LocationServices.API)
.addConnectionCallbacks(getServiceSetup())
.addOnConnectionFailedListener(getServiceFailureStrategy())
.build();
geofencingClient = LocationServices.getGeofencingClient(context);
Intent intent = new Intent(context, mIntentService);
// We use FLAG_UPDATE_CURRENT so that we get the same pending intent back when
// calling addGeofences() and removeGeofences().
geofencePendingIntent = PendingIntent.getService(context, 0, intent, PendingIntent.
FLAG_UPDATE_CURRENT);
geofencingClient.addGeofences(geofences, geofencePendingIntent)
.addOnSuccessListener(new OnSuccessListener<Void>() {
@Override
public void onSuccess(Void aVoid) {
getAddGeofencesCallback(runnable, geofencesToAdd).onResult(new Status(0));
Log.e(TAG, "Successfully added geofences with GeofencingClient");
}
})
.addOnFailureListener(new OnFailureListener() {
@Override
public void onFailure(Exception e) {
getAddGeofencesCallback(runnable, geofencesToAdd).onResult(new Status(1));
Log.e(TAG, "Failed to add geofences with GeofencingClient", e);
Intent 服务:
public class GeofenceService extends IntentService {
private static final String TAG = "GeofenceService";
public GeofenceService() {
super(TAG);
}
@Override
protected void onHandleIntent(Intent intent) {
GeofenceTransition transition = GeofenceServiceManager.getGeofenceTransition(intent);
if (transition == null) {
return;
}
GeofenceServiceManager.logd(
TAG,
"onHandleIntent. geofence: " +
GeofenceMessage.notificationTitleFor(this, transition)
);
processErrors(transition);
sendBroadcast(transition);
}
private Intent generateBroadcast(GeofenceTransition transition) {
Intent broadcastIntent = new Intent();
broadcastIntent.addCategory(GeofenceUtils.CATEGORY_GEOFENCE_SERVICES)
.setAction(GeofenceUtils.actionNameFor(transition))
.putStringArrayListExtra(
GeofenceUtils.GEOFENCE_IDS,
new ArrayList<String>(transition.getIds())
);
return broadcastIntent;
}
private void sendBroadcast(GeofenceTransition transition) {
if (transition.isError() || transition.unknownType()) {
GeofenceServiceManager.logd(TAG, "geofence transition is error or unknown type.");
return;
}
broadcastManager().sendBroadcast(generateBroadcast(transition));
}
private LocalBroadcastManager broadcastManager() {
return LocalBroadcastManager.getInstance(this);
}
private void processErrors(GeofenceTransition transition) {
if (!transition.isError()) {
return;
}
Log.e(TAG, "geofence error: "+GeofenceMessage.errorDetailsFor(this, transition));
Intent broadcastIntent = generateBroadcast(transition);
broadcastIntent.putExtra(
GeofenceUtils.GEOFENCE_STATUS,
GeofenceMessage.errorMessageFor(this, transition)
);
broadcastManager().sendBroadcast(broadcastIntent);
}
}
AndroidManifest.xml:
...
<uses-permission android:name="android.permission.ACCESS_FINE_LOCATION" />
<service
android:name=".geofence.GeofenceService"
android:enabled="true"
android:exported="false">
</service>
...
编辑 1: 根据 Google 记录的对 Android 8+ 后台处理的限制,这在我看来是不可能的。 Android 8+ 会在进入后台后 10 分钟内杀死正在运行的应用程序,并在不在前台时阻止启动服务,包括 Intent 服务。然而谷歌说:
Note: On Android 8.0 (API level 26) and higher, if an app is running in the background while monitoring a geofence, then the device responds to geofencing events every couple of minutes. To learn how to adapt your app to these response limits, see Background Location Limits.
鉴于已记录的后台执行限制,这怎么可能?
编辑 2:我已经看到 list 中声明的 BroadcastReceiver 可以使用 Intent 交付的蓝牙 LE 检测将应用程序启动到 Android 8+ 的后台。该代码如下所示:
Intent intent = new Intent();
intent.setComponent(new ComponentName(context.getPackageName(), "com.example.MyBroadcastReceiver"));
<receiver android:name="com.example.MyBroadcastReceiver">
</receiver>
这确实适用于 BLE 检测以将应用程序启动到后台。如果传送到 BroadcastReceiver,系统发起的 Intent 是否会以某种方式不受 Android 8+ 后台执行限制的影响(以 IntentServices 没有的方式)?如果是这样,这是否适用于地理围栏? (剧透警报:是的,它会!阅读下面接受的答案。)
最佳答案
你需要替换:
geofencePendingIntent = PendingIntent.getService(context, 0, intent, PendingIntent.
FLAG_UPDATE_CURRENT);
与
Intent intent = new Intent(this, GeofenceBroadcastReceiver.class);
mGeofencePendingIntent = PendingIntent.getBroadcast(this, 0, intent, PendingIntent.FLAG_UPDATE_CURRENT);
为了支持 Android 8,由于 Background Execution Limits,您应该使用 BroadcastReceiver 来接收地理围栏 Intent (服务不能在后台运行)
查看更多详细信息:https://github.com/android/location-samples/commit/5f83047c8a462d7c619f6275b624e219b4622322
google 示例如何使用地理围栏:https://github.com/googlesamples/android-play-location/tree/master/Geofencing
关于android - 背景地理围栏真的适用于 Android 8+ 吗?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/54949978/
我想知道地理数据、几何数据和数据库系统(例如 Oracle 或 SqlSqerver)中表示空间数据的标准数据类型之间有什么区别? 对我来说看起来是一样的,但我知道肯定有区别。. 最佳答案 首先,地理
我正在寻找用于存储 map 的理想数据库或数据结构的建议。本质上, map 由“道路”组成,如道路、路径等。道路包含节点(具有纬度和经度坐标,有时还有高度。) 任何此类数据库或结构: 应该能够快速(毫
我最近将一些预装的国家和省份形状下载到 Sql Server 2008 中 http://sqlsamplegeo.codeplex.com/ 除了地理列,每个国家和省都有一个“ShapeArea”和
我对google Analytics(分析)地理如何工作(特别是对于出差用户)有疑问。例如,如果我是森尼韦尔市的用户,并且我打开了一个应用程序,则Google ID在用户ID级别的分析将说用户数= 1
我正在使用 Geo::IP 对 IP 地址执行位置查找。一切正常,直到我遇到一个不在地理 IP 查找数据库中的 IP 地址并且程序突然关闭并给出此错误 Can't call method "city"
通用(或设备无关)物理位置的结构是什么?我的猜测是它可能是一个有两个长字段的结构,或者类似的东西。 此外,给定一个目的地位置和两个候选位置,是否有一种简单的算法可以确定哪个候选位置最接近目的地?我并不
我正在使用 d3 geomaps 创建美国的地理 map 。当我将鼠标悬停在状态上时,我希望突出显示该状态。那可能吗?这是我当前的代码:
在geoviews guide for Bokeh ,它指出 Bokeh 仅支持墨卡托投影,但大多数示例投影使用 PlateCarree,然后输出看起来像墨卡托投影。 有人知道图形的投影和输出是怎么回
我有一个geodataframe'all_locations',其中有一个几何列和一个带有点名称的列。在 map 上绘制点工作正常,但我想用位置名称注释这些点。 ['位置'] ['几何'] BUITH
我正在尝试在 Canvas 中缩放 map 。 var projection = d3.geoMercator() projection.fitExtent([[margin.left, margin
我目前正在处理这段代码,一切正常,但是当我用鼠标悬停在一个国家/地区时,我只希望国家/地区名称出现在标签中,而不是相关值。我可以这样做吗?如果是,怎么办? 提前致谢! 这是javascript代码:
我正在尝试在 mac os 上使用 Geodjango,我使用 postgresql 并安装了 GEOS,但出现此错误: dlopen(/usr/local/lib/libgeos_c.dylib,
我正在研究 GeoViews,我想知道我们是否可以将 slider 作为 GeoViews 中等值线图的输入。 我在 gdf 中有另一个变量,它是年份。是否可以使用 slider 显示年度 Tot
我有一个 Geometry (看起来像 WKB) 我的 postgres/postgis 数据库中的对象,我已经转储到 BigQuery 来做一些有趣的事情。我看到它存储为 bytes .我如何构建一
我正在使用 DPM 进行 azure 在线备份,并且想要将冗余从 Geo 更改为 Local,但它显示为灰色。有办法改变吗? 我不想为此创建新的错误,然后我必须再次将所有内容从 DPM 重新上传到 A
我正在实现 Google 地理 map ,我想做的是,例如,如果我单击 US在 map 上,我会将国家/地区名称作为警报(例如,如果我单击美国,我必须将美国作为警报),我最终可以将其用作变量。当我尝试
我正在制作一张具有缩放和平移功能的世界地图。我在某些城市上画了圆圈,圆圈的半径由数据决定。当鼠标悬停在这些圆圈上时,将出现一个工具提示来显示数据。 代码结构为 //在此选择上调用缩放行为 - 让我们调
我需要一个函数来计算一对 WGS 84 之间的距离定位到高精度,我计划使用 boost geometry 中的 geographic 函数. boost geometry Design Rationa
我正在尝试缓冲数据集中半径为 100 公里的点。我正在使用函数 gBuffer来自包裹rgeos .这是我到目前为止所拥有的: head( sampledf ) # postalcode
我正在用 C# 开发一个应用程序,它将使用 SQL Server 2008 中的“地理”数据类型来使用和存储地理位置。我计划使用 Entity Framework ,但很快发现它不支持空间数据。有没有
我是一名优秀的程序员,十分优秀!