5

我想我很好理解强关键字和弱关键字,但我不明白它在下面的代码中是如何使用的。此代码来自 Olivier Poitrey 的 SDWebImage,可在 github 上找到。我理解这里描述的强弱关键字:iOS5 中强弱存储的解释

下面的代码以一种让我好奇的方式使用了 __weak 和 __strong 关键字。它不是我习惯于看到弱使用的子父关系或委托模式。但是,我确信这是一种经常使用的模式,正如我之前在其他代码中看到的那样。它在另一个线程上运行的块之前设置一个 __weak 引用。然后,在块内,它将弱引用设置为强引用。

我确信这个好的和优雅的代码,所以我试图理解它。如果“self”在块运行之前不再存在,弱自引用将归零。当块运行时,强引用也将设置为零。因此,它会知道杀死其余的操作,因为 self 不再存在。我做对了吗?

现在,如果我们不使用 __weak 和 __strong 关键字会发生什么?如果我们只是在块内部检查 self == nil 会怎样。由于块复制整个树,“自我”永远不会为零吗?

有人可以帮助揭开这段令人敬畏的代码的神秘面纱吗?有人可以验证或否定我的假设吗?

- (void)setImageWithURL:(NSURL *)url placeholderImage:(UIImage *)placeholder options:(SDWebImageOptions)options progress:(SDWebImageDownloaderProgressBlock)progressBlock completed:(SDWebImageCompletedBlock)completedBlock;
{
    [self cancelCurrentImageLoad];

    self.image = placeholder;

    if (url)
    {
        __weak UIImageView *wself = self;
        id<SDWebImageOperation> operation = [SDWebImageManager.sharedManager downloadWithURL:url options:options progress:progressBlock completed:^(UIImage *image, NSError *error, SDImageCacheType cacheType, BOOL finished)
        {
            __strong UIImageView *sself = wself;
            if (!sself) return;
            if (image)
            {
                sself.image = image;
                [sself setNeedsLayout];
            }
            if (completedBlock && finished)
            {
                completedBlock(image, error, cacheType);
            }
        }];
        objc_setAssociatedObject(self, &operationKey, operation, OBJC_ASSOCIATION_RETAIN_NONATOMIC);
    }
}
4

2 回答 2

7

downloadWithUrl:方法可能需要很长时间。在那个时候,用户可能决定离开,消除对SDWebImage对象的需要。为了便于对象的早期清理,外部self引用是弱的。这样,downloadWithUrl不会阻止SDWebImage被释放

当然,如果你真的想使用self,你需要一个强有力的参考。因此,在完成块中downloadWithUrl获取对self. 如果对象在这段时间内消失,sself将会nil。否则,它将是一个有效的强引用,表明SDWebImage对象还在,此时对象将完成它的工作。

于 2013-02-16T07:35:47.460 回答
3

我确信这个好的和优雅的代码,所以我试图理解它。如果“self”在块运行之前不再存在,弱自引用将归零。当块运行时,强引用也将设置为零。因此,它会知道杀死其余的操作,因为 self 不再存在。我做对了吗?

不,你想太多了。__weak存储限定符就是这样:一个限定符。明确未保留的对象__weak,但如果从强变量分配,它们不仅会自动设置为 nil。事实上,这会破坏弱变量的目的!

现在,如果我们不使用 __weak 和 __strong 关键字会发生什么?如果我们只是在块内部检查 self == nil 会怎样。由于块复制整个树,“自我”永远不会为零吗?

检查实际上是不必要的,因为运行时将消息解析为 nil 到 nil(尽管如此,它可能对以后的实现很重要,谁知道呢)。你对这个很满意:如果没有那个“弱到强”的小舞,那么 self 将被块保留,有可能创造一个非常讨厌的保留循环。这就是我可以开始将这一切联系在一起的地方:

因为我们不希望块保留我们的变量,但我们也希望它在块的范围内是强的,所以不会发生任何奇怪的事情,将 self 分配给一个弱指针。当块发生在我们的弱指针上时,它不允许保留它,所以 self 的引用计数保持不变,然后一旦进入块,我们就回到一个强大的 self 变量,所以弱的变量被释放,我们不必再担心了。实际上,这意味着我们有一个可靠的保证,在整个块的执行过程中,self 要么是一个值,要么是 nil。很整洁吧?

于 2013-02-16T07:38:39.847 回答