ios - EXC_BAD_ACCESS 和泄漏可能是由于在 block 内捕获 block

标签 ios objective-c memory-leaks objective-c-blocks

我的应用最近在执行应用本身的核心功能之一时随机报告 EXC_BAD_ACCESS 崩溃,该错误一直存在,但在引入iOS 9.3.1。

我一直在使用 XCode 提供的工具(代码分析、NSZombies、Address Sanitizer 等)对代码进行大量分析和静态检查。在继续进行进一步调查之前,我想确保警告和潜在内存错误的计数降至 0。

现在我一直坚持这个警告“在此 block 中强烈捕获'endblock'可能会导致保留周期”。我真的认为解决这个警告也可以解决 EXC_BAD_ACCESS 问题,因为使用 Leaks 工具标有该警告的代码片段会导致大量泄漏。

这是导致泄漏的函数的代码片段:

- (void)arrayDayPlan:(void (^)()) block dateArray:(NSArray *)dateArray {

    [MyPlanner sharedInstance].multipleDaycounter = 1;
    NSInteger dayNumber = dateArray.count;

    void (^__block endBlock)() = ^void() {

        if([MyPlanner sharedInstance].multipleDaycounter == dayNumber) {
            block();
        } else {
            NSDate *newDate = [dateArray objectAtIndex:[MyPlanner sharedInstance].multipleDaycounter];
            [MyPlanner sharedInstance].multipleDaycounter = [MyPlanner sharedInstance].multipleDaycounter + 1;
            [[MyPlanner sharedInstance] plan:endBlock FromDay:[newDate dateAtStartOfDay] toDay:[[newDate dateByAddingDays:1] dateAtStartOfDay]];
        }
    };

    NSDate *firstDate = (NSDate *)[dateArray firstObject];
    [[MyPlanner sharedInstance] plan:endBlock FromDay:[firstDate dateAtStartOfDay] toDay:[[firstDate dateByAddingDays:1] dateAtStartOfDay]];
}

我收到警告的行是 block 内的这一行:

[[MyPlanner sharedInstance] plan:endBlock FromDay:[newDate dateAtStartOfDay] toDay:[[newDate dateByAddingDays:1] dateAtStartOfDay]];

虽然报告了此行的泄漏:

void (^__block endBlock)() = ^void() {

这个函数在项目的四个地方被调用,每个调用都或多或少具有以下结构:

- (void)planWithArray {

    [self showLoadingView];
    __block MyAssignedViewController *blockSafeSelf = self;

    [[MyPlanner sharedInstance] arrayDayPlan:^{

        [[MyPlanningData sharedInstance] resetDateToPlan];
        [blockSafeSelf refreshView];
        [blockSafeSelf dismissLoadingView];
    } dateArray:[[MyPlanningData sharedInstance] generateDateArray:[[NSDate now] dateAtStartOfDay]]];
}

我在 SO(herehere 和许多其他线程)上读到其他人也有类似的问题,但使用了自变量。它甚至适用于我的情况吗?

我还读到过使用 __weak 引用可以避免我遇到这个问题。我是否应该将 blockSafeSelf 定义从 __block 更改为 __weak

提前致谢!

最佳答案

错误:

Capturing 'endblock' strongly in this block is likely to lead to a retain cycle

与错误无关:

EXC_BAD_ACCESS

你确实有一个循环:

  • endblock 持有对 block 的引用;
  • 该 block 又持有对 endblock 的引用 - 由于 __block 限定符,它实际上是按引用捕获而不是按值捕获;和
  • __block 限定符是必需的,否则自引用将不起作用 - 如果 endblock 是按值捕获的,那么它的值将为 null,因为它的值为在分配给自身之前捕获。

因此,每次创建此 block 时,您都会可能泄漏一个 block 值,您可以通过创建阴影 __weak 变量并将其用于自引用来解决该问题,例如类似的东西:

typedef void (^EndBlock)(void);

EndBlock endBlock;
__block __weak _endBlock = endBlock = ^{
   ...
   // only reference _endBlock inside the block
   ...
};

// use endBlock outside the block itself

请注意,您仍然需要 __block 否则自引用将看到 nil(如果您不清楚自引用 block 发生了什么,请尝试 this answer from a few years ago ) .您还可以使用 __unsafe_unretained 而不是 __weak 如果您正在努力提高性能,它应该是安全的(如果有疑问,不要)。

但是,这样的泄漏不会导致您EXC_BAD_ACCESS - 相反,您往往会遇到此错误,因为某些东西在您期望的时候不存在,而不是在某些东西存在的时候当您不期望它时(泄漏)。

所以您需要到别处寻找您的EXC_BAD_ACCESS

HTH

关于ios - EXC_BAD_ACCESS 和泄漏可能是由于在 block 内捕获 block ,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/36667686/

相关文章:

ios - 如何生成或创建 iBeacons UUID?

ios - 使用 UIPickerView 上第一个组件的值更改其余组件

ios - Swift 4 - 带有 AVAudioPlayer 的 UISlider

ios - 使用 MPMoviePlayerController 时的内存泄漏

iphone - 从 UITextView 获取文本的来源

iphone - 要加倍的字符串

ios - NSArray removeObject 删除数组中的所有对象

iphone - 如何停止 UIScrollView 调用 drawRect : on sublayers that go in/out of the visible region?

javascript - 使用 NodeJS 遍历 mongo 数据流时内存泄漏

javascript - 在此 Windows 小工具中查找导致内存泄漏的原因