- iOS/Objective-C 元类和类别
- objective-c - -1001 错误,当 NSURLSession 通过 httpproxy 和/etc/hosts
- java - 使用网络类获取 url 地址
- ios - 推送通知中不播放声音
我的一个 iOS 应用程序似乎具有典型 Heisenbug 的症状。该应用程序跟踪用户的家庭位置,因此当用户进入和离开他们的家庭位置时会发生某些事件。
在我测试该应用程序时,它运行良好。我在 CLCircularRegion
中进进出出,无论我尝试哪种方式,它都能正常工作。它在后台与应用程序一起工作。它适用于关闭的应用程序。它与前台的应用程序一起工作。它适用于绿色鸡蛋和火腿。
不幸的是,用户报告会延迟 15 分钟左右的问题。用户将进入他们的家,但事件要等到以后才会发生。在某些情况下,事件根本不会发生。该模式似乎是当用户第一次开始使用该应用程序时,它运行良好。大约一天后,该应用程序似乎无法正常运行。事件延迟。
我会首先承认我不是 CLLocationManager
和 CLCircularRegion
内部工作的专家。我相信我已经正确设置了一切,但我真的很难弄清楚如何调试这样的东西。
无论如何,我会在这里展示我的一些代码。请记住,这是使用 Xamarin 开发的,因此它在 C# 中。
AppDelegate.cs
public static AppDelegate self;
private CLLocationManager locationManager;
private CLCircularRegion[] locationFences;
private void initializeLocationManager()
{
this.locationManager = new CLLocationManager();
// iOS 8 additional permissions requirements
if (UIDevice.CurrentDevice.CheckSystemVersion(8, 0))
{
locationManager.RequestAlwaysAuthorization();
}
locationManager.AuthorizationChanged += (sender, e) =>
{
var status = e.Status;
// Location services was turned off or turned off for this specific application.
if (status == CLAuthorizationStatus.Denied)
{
stopLocationUpdates();
}
else if (status == CLAuthorizationStatus.AuthorizedAlways &&
iOSMethods.getKeyChainBool(OptionsViewController.GENERIC, OptionsViewController.SERVICE_GEOLOCATION_ENABLED))
{
startLocationUpdates();
}
};
if (CLLocationManager.IsMonitoringAvailable(typeof(CLCircularRegion)))
{
locationManager.RegionEntered += (sender, e) =>
{
setRegionStatus(e, "Inside");
};
locationManager.RegionLeft += (sender, e) =>
{
setRegionStatus(e, "Outside");
};
locationManager.DidDetermineState += (sender, e) =>
{
setRegionStatus(e);
};
}
else
{
// cant do it with this device
}
init();
}
public void init()
{
var data = SQL.query<SQLTables.RoomLocationData>("SELECT * FROM RoomLocationData").ToArray();
int dLen = data.Length;
if (dLen > 0)
{
locationFences = new CLCircularRegion[dLen];
for (int x = 0; x < dLen; x++)
{
var d = data[x];
CLCircularRegion locationFence = new CLCircularRegion(new CLLocationCoordinate2D(d.Latitude, d.Longitude), d.Radius, d.SomeID.ToString() + ":" + d.AnotherID.ToString());
locationFence.NotifyOnEntry = true;
locationFence.NotifyOnExit = true;
locationFences[x] = locationFence;
}
}
}
private void setRegionStatus(CLRegionEventArgs e, string status, bool calledFromDidDetermineState = false)
{
string identifier = e.Region.Identifier;
string lastStatus = iOSMethods.getKeyChainItem(OptionsViewController.GENERIC, OptionsViewController.SERVICE_LAST_GEO_STATUS);
if (lastStatus == status + ":" + identifier)
{
return;
}
iOSMethods.setKeychainItem(OptionsViewController.GENERIC, OptionsViewController.SERVICE_LAST_GEO_STATUS, status + ":" + identifier);
string[] split = identifier.Split(new string[] { ":" }, StringSplitOptions.RemoveEmptyEntries);
if (split.Length == 2)
{
try
{
int someID = Convert.ToInt32(split[0]);
int anotherID = Convert.ToInt32(split[1]);
// Notifies our API of a change.
updateGeofenceStatus(someID, anotherID, status);
if (iOSMethods.getKeyChainBool(OptionsViewController.GENERIC, OptionsViewController.SERVICE_GEOLOCATION_NOTIFICATIONS) &&
(status == "Inside" || status == "Outside" || status == "Unknown"))
{
var rm = SQL.query<SQLTables.KeyRoomPropertyData>("SELECT * FROM KeyRoomPropertyData WHERE SomeID ID = ? AND AnotherID = ?",
new object[] { someID, anotherID }).ToArray();
if (rm.Length > 0)
{
if (status == "Unknown")
{
return;
}
var rmD = rm[0];
UILocalNotification notification = new UILocalNotification();
notification.AlertAction = "Geolocation Event";
notification.AlertBody = status == "Inside" ? "Entered " + rmD.SomeName + ": " + rmD.AnotherName :
status == "Outside" ? "Exited " + rmD.SomeName + ": " + rmD.AnotherName :
"Geolocation update failed. If you would like to continue to use Geolocation, please make sure location services are enabled and are allowed for this application.";
notification.SoundName = UILocalNotification.DefaultSoundName;
notification.FireDate = NSDate.Now;
UIApplication.SharedApplication.ScheduleLocalNotification(notification);
}
}
}
catch (Exception er)
{
// conversion failed. we don't have ints for some reason.
}
}
}
private void setRegionStatus(CLRegionStateDeterminedEventArgs e)
{
string state = "";
if (e.State == CLRegionState.Inside)
{
state = "Inside";
}
else if (e.State == CLRegionState.Outside)
{
state = "Outside";
}
else
{
state = "Unknown";
}
CLRegionEventArgs ee = new CLRegionEventArgs(e.Region);
setRegionStatus(ee, state, true);
}
public void startLocationUpdates()
{
if (CLLocationManager.LocationServicesEnabled)
{
init();
if (locationFences != null)
{
foreach (CLCircularRegion location in locationFences)
{
locationManager.StartMonitoring(location);
Timer t = new Timer(new TimerCallback(delegate(object o) { locationManager.RequestState(location); }), null, TimeSpan.FromMilliseconds(500), TimeSpan.FromMilliseconds(-1));
}
}
}
}
public void stopLocationUpdates(bool isRestarting = false)
{
if (locationFences != null)
{
foreach (CLCircularRegion location in locationFences)
{
locationManager.StopMonitoring(location);
}
}
if (!isRestarting)
{
var rooms = SQL.query<SQLTables.KeyRoomPropertyData>("SELECT * FROM KeyRoomPropertyData").ToArray();
foreach (SQLTables.KeyRoomPropertyData room in rooms)
{
// notifies our API of a change
updateGeofenceStatus(room.SomeID, room.AnotherID, "Unknown");
}
}
}
我知道有很多代码可供任何人筛选,但目前我真的没有很好的理论来说明是什么导致了这个错误,或者是否有可能修复 iOS 的限制。
我的一些理论是,如果 CLLocationManager
.PausesLocationUpdatesAutomatically
属性可能与它有关,或者 CLLocationManager
的一些其他属性> 例如 ActivityType
、DesiredAccuracy
或 DistanceFilter
。我将所有这些都保留为默认值,我认为这没问题,但我不太确定。
另一种说法是,“服务”在后台运行了一段时间后,有一个未捕获的异常被抛出。如果是这样的话,iOS 有什么可以给我堆栈跟踪或其他东西的吗?在我所有的测试中,我从未遇到过从这段代码中抛出的任何异常,所以我有点怀疑这就是问题所在。不过在这一点上,我愿意接受任何想法或建议。
另外,请记住,为了让这个应用程序按预期方式工作,位置更新事件必须在用户进入或离开 CLCircularRegion
时立即发生(在一分钟内至少如此)。显然,我必须让用户保持启用他们的位置服务并允许应用程序拥有适当的权限。
最佳答案
您的诊断很可能是正确的 - 这是典型的观察者效应。
当您测试应用程序时,当用户玩新应用程序时,iphone 正在被积极使用。它没有机会休眠。第二天,当用户返回家时会发生什么——他们的手机很可能在到达家之前的很长一段时间内没有被使用:通常我们在离开公共(public)交通工具后的“最后一英里”步行过程中或开车回家时不使用手机家。 iOS 会注意到这种延长的不活动时间并调整自己的行为以优化电池生命周期。
观察这一点的最简单方法是组合一个简单的面包屑应用程序 - 在您的位置设置地理围栏,并在每次收到退出事件时继续这样做。根据您使用(或不使用)的方式,走同一条路线时您的手机结果会大不相同。
当您回到家时,手机通常也是您最不可能拿到的东西。
您可能希望用户提供更多详细信息,包括他们在回家前后 15 分钟内使用手机的具体情况、他们使用的其他应用程序、如果他们开车,他们是否会继续运行导航应用程序等。您将找出规律。
re. Also, please keep in mind that in order for this application to work the way it was intended, the location update events MUST occur as soon as the user enters or exist the CLCircularRegion (within a minute or so at least).
您不能只使用地理围栏来做到这一点,尤其是考虑到不同的到达/离开模式 - 步行与驾驶、“下降”路径(例如掉头到达)。您必须预料到超过 1 分钟的延迟和“过早”触发。恐怕没有解决方法。
关于c# - iOS 跟踪 CLCircularRegion - Heisenbug,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/35134729/
我正在开发一个应用程序,该应用程序从使用 performSelectorInBackground 启动的单独线程中的循环调用 AudioServicesPlayAlertSound(kSystemSo
如果我在循环中添加调试输出,我有一个行为不同的循环。我想知道这是否是一个编译器错误,或者我是否错误地依赖了一些未定义的 C++ 行为。 循环从 uint64_t 数组中读取整数,将它们存储在临时数组中
我在域套接字上调用 async_read,我的问题是有时我会获得数据,有时我没有,即使远程系统总是返回数据。如果我进行逐步调试,我似乎能够从套接字中读取。如果我自动化单元测试,它们似乎运行得太快而无法
经过一些资源修改(图像)后,我的程序随机崩溃 (0xC0000005)。我知道这些来自糟糕的内存管理,但我找不到它的来源,原因有二:首先,我没有使用很多指针或手动动态分配的变量(我的意思是我使用的是标
好的,我有一个非常奇怪的 CSS 问题,我想知道任何人都可以提出解释。 重现步骤: 打开 Chrome 并导航到 http://www.mcwhinneys.com/media照片库应该没有对齐,向右
好的 StackOverflow - 这是一个奇怪的。 问题 所以我有一个“按钮”(实际上只是一个带有 javascript onclick 监听器的 div),它在页面加载时通过 json 从数据库
我最近修复了我们产品中的一个缺陷,其症状是访问悬空指针导致的访问冲突。 为了良好的实践,我添加了一个单元测试以确保错误不会再次出现。在编写单元测试时,我总是会撤消我的缺陷修复并确保单元测试失败,否则我
使用Django 1.4/Python 2.7/reportlab(开源版)生成pdf。 到目前为止,一切都非常顺利。以前的 pdf 生成(如在 http 中请求返回/下载生成的 pdf 文件)在 D
我刚刚花了最后一个小时来解决 C# 中非托管内存的一个奇怪问题。 首先,一些上下文。我有一个 C# DLL,它导出一些 native 方法(通过 this awesome project templa
我想注意的是,我将描述的场景很少发生,并且在大多数情况下,一切都按预期进行。 我在 Pub/Sub 端有 1 个主题和 1 个订阅。 我的 java 应用程序监听订阅,进行一些处理并发送回确认。由于g
我遇到了一个非常令人困惑的崩溃,我目前正在用尽我的智慧...... 首先是崩溃日志: 日期/时间:2012-02-14 10:55:09.771 +0100 操作系统版本:Mac OS X 10.7.
我正在开发基于 WebKit(使用 C++/Qt4)并支持 JavaScript 的 headless 浏览器。这样做的主要目的是能够生成大量基于 JavaScript 的网站的 HTML 快照(请参
我的一个 iOS 应用程序似乎具有典型 Heisenbug 的症状。该应用程序跟踪用户的家庭位置,因此当用户进入和离开他们的家庭位置时会发生某些事件。 在我测试该应用程序时,它运行良好。我在 CLCi
我有一个 Heisenbug 的经典示例,它是由我以前从未见过的条件触发的。我的遗留应用程序(大约 100K sloc 的旧代码)在特定实例中无法正常工作,仅仅启用 JPDA 远程调试足以改变行为,导
我已无路可走:我有一个单线程 C++ 程序。这里有一些经验数据和背景信息,我试图突出最重要的关键词; 我所说的整个部分没有任何系统调用,除了标准 C++ 库 可能执行的内存(取消)分配调用(std::
在开发 Spring boot REST 端点期间,我的应用程序遇到了奇怪的(heisenbug)行为。也可能与我为每个端点项目制作单独的模块有关。具体来说,它可以运行一次但重新运行后会失败,可能运行
我称此为 Heisenbug 错误,因为该错误仅在未观察到时才存在。这是调用: @selected_members = Member.where(id: params[:member_ids]) @
我正在开发一个使用压控振荡器芯片 (VCO) 来帮助处理信号的系统。芯片制造商 (Analog Devices) 提供了一个将设置文件加载到 VCO 上的程序,但我希望能够从总体信号处理控制系统中设置
最近,我遇到了一个经典的 Heisenbug .情况是这样的: 我在一个面板中有一个树列表,即主视图,在右侧的另一个面板中有一个详细 View ,显示有关当前选定树节点的信息。 (与 Windows
我在 Java 6/Websphere 8.5 上遇到套接字绑定(bind)错误(Liberty 配置文件,Websphere 的精简可用版本)。当再次立即杀死并启动应用程序服务器时,我得到: [ER
我是一名优秀的程序员,十分优秀!