- android - RelativeLayout 背景可绘制重叠内容
- android - 如何链接 cpufeatures lib 以获取 native android 库?
- java - OnItemClickListener 不起作用,但 OnLongItemClickListener 在自定义 ListView 中起作用
- java - Android 文件转字符串
我最近在一些Objective-C代码中偶然发现了The Deallocation Problem。之前在Block_release deallocating UI objects on a background thread中的Stack Overflow上讨论了该主题。我想我了解这个问题及其含义,但是可以肯定的是,我想在一个小的测试项目中重现它。我首先创建了自己的SOUnsafeObject
(=应该始终在主线程上释放的对象)。
@interface SOUnsafeObject : NSObject
@property (strong) NSString *title;
- (void)reloadDataInBackground;
@end
@implementation SOUnsafeObject
- (void)reloadDataInBackground {
dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
dispatch_async(dispatch_get_main_queue(), ^{
self.title = @"Retrieved data";
});
sleep(3);
});
}
- (void)dealloc {
NSAssert([NSThread isMainThread], @"Object should always be deallocated on the main thread");
}
@end}]
[[[SOUnsafeObject alloc] init] reloadDataInBackground];
放入
application:didFinishLaunching..
中,则该应用会在3秒后由于断言失败而崩溃。建议的修复程序似乎有效。 IE。如果我将
reloadDataInBackground
的实现更改为:该应用程序不再崩溃:
__block SOUnsafeObject *safeSelf = self;
dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
dispatch_async(dispatch_get_main_queue(), ^{
safeSelf.title = @"Retrieved data";
safeSelf = nil;
});
sleep(3);
});
UIViewController
进行相同的尝试(因为
UIViewController
可能会在现实生活中扮演
SOUnsafeObject
的角色)。实现几乎与
SOUnsafeObject
相同:
@interface SODemoViewController : UIViewController
- (void)reloadDataInBackground;
@end
@implementation SODemoViewController
- (void)reloadDataInBackground {
dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
dispatch_async(dispatch_get_main_queue(), ^{
self.title = @"Retrieved data";
});
sleep(3);
});
}
- (void)dealloc {
NSAssert([NSThread isMainThread], @"UI objects should always be deallocated on the main thread");
NSLog(@"I'm deallocated!");
}
@end
[[SODemoViewController alloc] init] reloadDataInBackground];
放入
application:didFinishLaunching..
。嗯,断言不会失败。.消息
I'm deallocated!
在3秒钟后打印到控制台,因此我很确定 View Controller 将被释放。
UIViewController
始终释放在主线程上?我开始怀疑这一点,因为以下代码片段也不会破坏我的断言:
dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), {
SODemoViewController()
});
__weak
引用的事实,但是让我们假设 View Controller 应该仍然活着,可以在主线程上执行我们的完成代码。另外,在尝试解决此问题之前,我正在尝试了解问题的核心。我将代码转换为Swift,并获得了与Objective-C相同的结果(
SOUnsafeObject
的修复在语法上更加丑陋)。
最佳答案
tl;博士-虽然我找不到官方文档,但是当前的实现确实确保了dealloc
的UIViewController
在主线程上发生。
我想我只能给出一个简单的答案,但是也许今天我可以做些“教人钓鱼”的事情。
好的。我在任何地方都找不到此文档,而且我也不记得曾经公开说过它。实际上,我始终不遗余力地确保将 View Controller 分配在主线程上,这是我第一次看到有人指出UIViewController
对象自动分配在主线程上。
也许其他人可以找到一份正式声明,但我找不到。
但是,我确实有证据证明确实如此。实际上,起初,我以为您没有正确处理块或引用计数,并且某种程度上在主线程上保留了引用。
但是,经过粗略的看后,我很有兴趣亲自尝试一下。为了满足我的好奇心,我制作了一个类似于您的从UIViewController
继承的类。它的dealloc
运行在主线程上。
因此,我只是将基类更改为UIResponder
,它是UIViewController
的基类,然后再次运行它。这次,它的dealloc
在后台线程上运行。
嗯也许闭门造车。我们有很多调试技巧。答案总是取决于您尝试的最后一个方法,但是我认为我会尝试使用通常的技巧来解决此类问题。
日志通知
找出所有实现方式的我最喜欢的工具之一是记录所有通知。
[[NSNotificationCenter defaultCenter]
addObserverForName:nil
object:nil
queue:nil
usingBlock:^(NSNotification *note) { NSLog(@"%@", note); }];
instrumentObjcMessageSends(YES);
dealloc
方法的第一行。
instrumentObjcMessageSends(NO);
extern void instrumentObjcMessageSends(BOOL);
$ cat msgSends-72013
- __NSMallocBlock__ __NSMallocBlock release
- SOUnsafeObject SOUnsafeObject dealloc
$ cat msgSends-72057
- __NSMallocBlock__ __NSMallocBlock release
- SOUnsafeObject UIViewController release
- SOUnsafeObject SOUnsafeObject dealloc
UIViewController release
的存在表明
UIViewController
对
+release
方法具有特殊的替代实现。我想知道为什么?可以专门将对
dealloc
的调用转移到主线程吗?
UIViewController
中存在替代,因此我可以正常进行操作。我发现当我跳过步骤时,通常需要更长的时间。
UIViewController
。
(lldb) b [UIViewController release]
Breakpoint 3: where = UIKit`-[UIViewController release], address = 0x000000010e814d1a
pthread_main_np
是一个告诉您是否在主线程上运行的函数。逐步执行汇编指令,确认我们没有在主线程上运行。
dealloc
的调用,而是运行您可以轻松看到的代码,以便在主线程上运行dealloc-helper。
release
(如果在禁用ARC的情况下进行编译),并且如果它们不调用基类实现,则不会出现此行为。
关于ios - 为什么将UIViewController释放在主线程上?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/32006565/
我希望能够在代码中布局我的 View Controller ,但可以看到界面构建器中显示的布局。 我知道我可以创建一个 UIView 子类,使其 IBDesignable,并将其分配给 View Co
我有一个父 UIViewController(MainVC)。从那里我有 2 个 segue 到 2 个 UIViewControllers:FirstVC(标识符:goFirstVC)和 Secon
我有一个导航 Controller 。第一个viewcontroller 是FirstViewController 的一种。当我在 FirstViewController 中点击一个按钮时,它会将第二
我有一个自定义的动画 UIViewController 过渡,似乎 iOS 中有一个错误会破坏横向布局。在主要动画方法中,我得到了横向和纵向 View 的混合。 (纵向 View 都是纵向的,所以没问
很难说出这里要问什么。这个问题模棱两可、含糊不清、不完整、过于宽泛或夸夸其谈,无法以目前的形式得到合理的回答。如需帮助澄清此问题以便重新打开,visit the help center . 关闭 9
我有一个 UIViewController,里面有 UITabBar。我正在尝试模仿 UITabBarController。 My question is how do I set or a UIVi
是否有任何方便的方法来确定是否正在从处于后台模式的应用加载 View ? 在 3.X 中,我将依赖 viewDidLoad 进行一些初始化等操作,但是对于 4.X 而言情况并非如此,因为您不能依赖调用
我将主视图 Controller 作为带有 UICollectionView 的 View Controller ,并且填充了照片。 一旦一个单元格被点击,一个新的 View Controller 被
我有一个简单的 UIViewController(单点触控),在 UI 上有一个按钮。现在,我的用例很简单 - 我想在单击按钮时重新加载 UIViewController。在某些线程上,我读到我只需要
我在它上面有一个 UIViewController 1 UIButton(添加为 subview ),在我按下 Button(见下面的图 1)之后,在它上面添加了另一个 UIViewControlle
尝试为 iPad 制作基于 Storyboard的应用程序。在其中,我需要从开始屏幕 (UIViewController) 过渡到主屏幕 (UISplitViewController),然后再到全屏
我想调整 UIViewController 的大小在 Storyboard中,所以我可以使用它的 UIViewController作为弹出 View 。 在大多数网站上,我可以读到这些操作是: 拖一个
我正在尝试从 XIB 或 Storyboard加载 ViewModel 绑定(bind)的 ViewController(使用 MvvmCross 5.0.6)。我的应用程序的逻辑是,该 View C
我正在构建一个锻炼应用程序。我的 TableViewController 中有一系列练习,每个单元格显示不同的练习。单元格转到 UIViewController。在 UIViewController
我正在尝试从 XIB 或 Storyboard加载 ViewModel 绑定(bind)的 ViewController(使用 MvvmCross 5.0.6)。我的应用程序的逻辑是,该 View C
我有一个 UIViewController,它有一个模式窗口,我想在整个界面上展示它,包括 UITabBar。 我的应用程序层次结构是这样的: UITabBarController (A) ->
我想显示另一个 ViewController 的缩小“预览”在另一个 ViewController .这可能吗?不需要交互,因此它基本上可以是屏幕截图,但按比例缩小。我可以通过手动截屏并将其保存为图像
我正在开发一个应用程序,其中 UIViewController ( firstViewController ) 包含一些 UILabels , 一个 UIButton , 和 UIView ( sub
我的标签栏中有五个标签栏项目。第五项是弹出ViewController。因此,当我单击该按钮时,当前 Controller 中将显示一个弹出窗口。我使用 UIViewController 作为子类来实
我正在开发 iOS 应用程序。 我的应用程序有 6 个按钮,每个按钮都有每个标签栏控件。 so Main screen is [btn1] [btn2] [btn3] [btn4] [btn5] [b
我是一名优秀的程序员,十分优秀!