- android - RelativeLayout 背景可绘制重叠内容
- android - 如何链接 cpufeatures lib 以获取 native android 库?
- java - OnItemClickListener 不起作用,但 OnLongItemClickListener 在自定义 ListView 中起作用
- java - Android 文件转字符串
我们的应用程序有一个类试图实现 Receptionist Pattern for KVO observation .整个应用程序中的其他类(例如 View Controller )创建这个 Receptionist 类的实例以充当 KVO 观察者。每个 Receptionist 实例都保留所有者提供的 block 的副本,当 KVO 通知到达时,Receptionist 实例将在适当的操作队列上调用该副本。
接待员的 dealloc 方法调用 KVO removeObserver 方法。 Owner 将 Receptionist 实例保留为强引用字段,因此当 Owner 被释放时,Receptionist 将在释放过程中将自己作为观察者移除。
当 Receptionist 实例在一个线程上收到 KVO 通知,而同一实例的 dealloc 正在另一个线程上进行时,我们看到了来自崩溃领域的报告。接待员的 observeValueForKeyPath:ofObject:change:context: 实现在这一行崩溃:
__weak typeof(self) weakSelf = self;
崩溃报告中的堆栈跟踪显示这是对 objc_initWeak 的调用,后者调用 weak_register_no_lock,后者调用 _objc_fatal。
这个特定的接待员正在观察其键的对象永远不会被释放。 Owner 也没有被解除分配;所有者正在用另一个接待员实例替换此接待员实例。
我能理解为一个已经被释放的对象创建一个弱引用是没有用的,但我希望 weakSelf 接收到一个 nil 值,而不是导致崩溃。
documentation for objc_initWeak如果所需引用的参数已开始释放,则明确提及将目标设置为 null。这听起来像是理想的行为,但我认为这不是我所看到的。我不希望用对 objc_initWeak 的显式调用来替换该行,因为我怀疑我能否正确管理释放。
在请求对自身的弱引用之前,接待员真的有责任注意到它自己的释放正在进行中吗?我假设在 NSObject 的释放开始和调用该对象的 dealloc 方法之间存在一些时间窗口,因此 dealloc 方法在对象内发出的信号听起来不稳定。
感谢阅读!
PS:在阅读了 Ken Thomases 提出的问题后进行了大量编辑。
最佳答案
这与弱引用的创建无关。您引用的行只能在对 self
有强烈引用的上下文中运行。
想一想:您看到的崩溃可能发生在您的 observeValueForKeyPath:ofObject:change:context:
实现中的那一行,但是,因为在释放和调用之间显然存在竞争对于该方法,释放也可能发生在该方法调用的分派(dispatch)期间(或其他某个点)。你很容易受到不同的崩溃。因此,对方法的实现进行任何更改都不可能解决问题,因为问题甚至可能在您的方法被调用之前就已经显现。
如果您要在该对象上调用方法,则您有责任保持对该对象的强引用。或者,从另一个角度来看,避免在您不确定在调用期间是否存在的对象指针上调用方法(因为您持有强引用或其他一些 API 保证)。
使用 KVO,您需要在发布最后一个强引用之前移除观察者。
关于ios - 为什么在 dealloc 发生在另一个线程上时尝试创建对 self 的弱引用时我会崩溃?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/47561938/
我一直在尝试编写我自己的弱/强指针,但我并不清楚其中的关系。我似乎遇到的所有事情都没有说清楚,而且一个医生经常会反驳另一个医生所说的话。任何人都可以详细解释弱/强指针关系,也许还有图像或代码示例吗?
静态/动态和强/弱类型之间有什么区别? 最佳答案 静态/动态类型涉及何时获取类型信息(在编译时或运行时) 强/弱类型是关于如何严格区分类型(例如,语言是否尝试从字符串到数字进行隐式转换)。 请参阅wi
我有一个非常奇怪的情况。我的服务器当前已关闭并收到 503 http 状态代码。基于如下给定的代码,代码进入 if 条件,但是当我将调试点置于 let error = self?.decodeErro
对于短期运行的操作,避免[weak self]是否可以接受?例如,URLSession 将保留 dataTask(with:completion:) 的闭包: final class ViewCont
我有一个非常奇怪的情况。我的服务器当前已关闭并收到 503 http 状态代码。基于如下给定的代码,代码进入 if 条件,但是当我将调试点置于 let error = self?.decodeErro
假设我有以下情况: Test1.java import java.lang.ref.WeakReference; public class Test1 { public WeakReferen
有没有办法告诉模拟器(我正在使用 Modelsim)当信号不是由任一双向接口(interface)驱动时将信号拉到弱“H”? 例如,如果我有一个 I2C 信号 I2C_SDA 被声明为来自 2 个模块
这是将一些值放入 WeakHashMap 中然后从映射中删除这些值的代码片段。它如何处理分配的内存? import java.util.*; public class WeakHashMap_Main
我正在尝试弄清楚智能指针可以实现什么。 但有一些感觉像是障碍。 普通指针有一个简短的定义 Someclass *p但是智能指针有点长shared_ptr p当您必须处理这些指针的模板(如 vector
这两行代码有区别吗? __weak IBOutlet UITextField *usernameField; @property (weak) IBOutlet UITextField *userna
我最近发现了 WeakHashMap Java 中的数据结构。 但是,我不明白它在不再正常使用时对映射进行垃圾收集是什么意思。数据结构如何知道我将不再在我的程序中使用 key ?如果长时间不引用 ke
我的问题是为什么 weak IBOutletCollection 总是 nil?如果将弱变强,我所有的按钮都在那里,这真的很奇怪。我试图理解苹果的逻辑,我看不出单个按钮和一组按钮在内存管理方面没有区别
我创建一个 WeakHashMap 为 WeakHashMap map = new WeakHashMap(); map.put(emp,"hello"); 其中 emp 是一个 Employee 对
在delphi sydney中,在对象(不是接口(interface))前面设置[weak]会受到惩罚吗?示例: TMyObject = class(Tobject) Private
在delphi sydney中,在对象(不是接口(interface))前面设置[weak]会受到惩罚吗?示例: TMyObject = class(Tobject) Private
众所周知,我们将声明一个可以打破强引用循环的弱委托(delegate)对象: // MyObject.h ... @property (nonatomic, weak) id delegate; ..
我已阅读this article关于Java中不同类型的引用(强引用、软引用、弱引用、幻像引用),但我不太理解。 这些引用类型之间有什么区别?每种类型何时使用? 最佳答案 Java 提供了两种不同类型
我突然想到...我相信弱引用的生命 与该引用的范围(在函数内或全局内)相关。 所以我想知道,只要我将数据处理保持在特定范围内,那么我应该可以使用 weak 与 strong 引用。正确的? 我问的原因
func addAdditionalElement(_ additionalSelectedElementsIDs: [String], startX: CGFloat, containerView:
我想要一个指针,以便我可以判断引用计数何时为 1。本质上,指针的工作方式类似于 weak_ptr,但清理工作需要手动进行。也就是说,程序每隔一段时间就会经历一个指针循环,并检查哪些指针只剩下一个引用。
我是一名优秀的程序员,十分优秀!