objective-c - 为什么复制指针以解决 ARC 下基于 block 的保留循环是有意义的?

标签 objective-c ios memory-management automatic-ref-counting objective-c-blocks

例如,在 ARC 下,如果您在 block 内使用 self,则怀疑 block 会导致保留循环。

我见过 workaround here , 像这样: enter image description here

此解决方法如何防止保留周期?

weakRequest 只是指向由 request 引用的完全相同对象的指针。当 ARC 修改 weakRequestrequest 的保留计数时,它会影响同一个对象。

然后,在 block 中,发生了一件奇怪的事情:

__strong ASIHTTPRequest *strongRequest = weakRequest;

这相当于说:

ASIHTTPRequest *strongRequest = weakRequest;
[strongRequest retain];

但再次强调:这是同一个对象。为什么所有这些不同的变量名称?它们只是指针!

我从来没有真正关心过障碍并试图避免它们。但是现在这让我很好奇当他们说“一个 block 捕获变量”时,每个人都在谈论什么。直到今天,我认为这只是意味着一个 block 将保留您使用的每个在 block 范围之外定义的指针,这意味着该 block 只保留您在其范围内触摸的任何对象。

我做了这个快速测试:

UIView *v = [[UIView alloc] initWithFrame:CGRectMake(100, 100, 100, 100)];
[self.view addSubview:v];
v.backgroundColor = [UIColor orangeColor];

NSLog(@"self = %p", self); // 0x6a12a40

[UIView animateWithDuration:1.5 
                      delay:0
                    options:UIViewAnimationOptionAllowUserInteraction
                 animations:^{
                     UIViewController *my = self;
                     NSLog(@"my = %p", my); // 0x6a12a40
                     v.frame = CGRectMake(200, 200, 100, 100);
                 }
                 completion:nil];

就像您看到的那样,对象本身保持完全相同。该 block 不会创建副本。所以我可以放心地假设多年的 C 和 Objective-C 知识仍然有效:

ASIHTTPRequest *strongRequest = internetRequest;
ASIHTTPRequest *foo = strongRequest;
ASIHTTPRequest *bar = foo;

if (bar == internetRequest) {
    NSLog(@"exact same thing, of course");
}

那么那里发生了什么?如果所发生的只是创建指向同一对象的不同指针,这如何解决保留计数?为什么要额外创建这些指针?

这不是完全一样吗?

[request setCompletionBlock:^{
    NSString *respondeString = [request responseString];
    if ([_delegate respondsToSelector:@selector(pingSuccessful:)]) {
        [_delegate pingSuccessful:responseString];
    }
}];

关于 Objective-C 一定有一些 secret 可以解释为什么复制指针在这里解决了内存管​​理问题。这对我来说没有任何意义。

最佳答案

它实际上与ARC无关,而是 block 如何捕获变量。指针被复制,以便 block 捕获的变量具有正确的所有权限定符。

weakRequest is just a pointer to the exact same object referenced by request. When ARC modifies the retain count of weakRequest or request, it's affecting the same object.

是的,它们都指向同一个对象,但是 weakRequest__unsafe_unretained 所有权限定符,这意味着当该变量被 block 捕获时,它的保留计数不变。

如果 request 被 block 捕获,那么无论您是否使用 ARC,它都会被保留并且您将拥有保留周期。

转换回 __strong 指针只是让该对象在阻止执行期间保持事件状态。

关于objective-c - 为什么复制指针以解决 ARC 下基于 block 的保留循环是有意义的?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/8857281/

相关文章:

ios - 使用AFNetworking _convertJSONString保持分配状态

objective-c - 应用程序终止时泄漏

c - C 存储变量的高级内存管理

ios - 如何检测像 Maps.app 这样的 MKPolylines/Overlays 上的点击?

objective-c - 将 NSTitlebarAccessoryViewController 添加到 NSWindow 后出现延迟

ios - 同时更改所有表格 View 单元格上的显示参数

ios - 滚动表格 View 时,标签的角半径变得扭曲

带有标签栏和导航 Controller 的 iOS 应用

ios - 链接 `NSOperation` : Pass result from an operation to the next one

objective-c - iOS 中调用方法有什么区别