草庐IT

ios - 从数组访问 NSNumber 时崩溃

coder 2024-01-22 原文

我遇到了一个奇怪的崩溃,其中一个 NSNumber 的实例似乎被释放了,尽管它仍然存在于数组中。我创建了一个系统来从远程服务器下载多个文件,并有一个 block 来指示进度(实际上是平均进度)。并且进度的计算产生了崩溃。崩溃不一致,并且“通常”发生在任何时候都可能发生。 [_NSProgressFractionTuple floatValue]:无法识别的选择器发送到实例 0x17042ab80 让我相信 NSNumber 以某种方式被释放,但我看不出这是怎么可能的。

给出完整的方法代码:

- (void)downloadFilesFromURLs:(NSArray<NSString *> *)urlPaths withProgress:(void (^)(float progress))progressBlock completion:(void (^)(NSError *error))completionBlock {
    NSMutableArray *toLoad = [[NSMutableArray alloc] init];
    for(NSString *path in urlPaths) {
        if([self fileForURL:path] == nil) {
            [toLoad addObject:path];
        }
    }

    NSInteger itemsToLoad = toLoad.count;

    if(itemsToLoad <= 0) {
        if(completionBlock) {
            completionBlock(nil);
        }
        return;
    }



    // Set progresses to zero
    __block NSMutableArray *progresses = [[NSMutableArray alloc] init];
    for(int i=0; i<itemsToLoad; i++) [progresses addObject:[[NSNumber alloc] initWithFloat:.0f]];

    __block NSInteger requestsOut = itemsToLoad;
    __block NSError *latestError = nil;
    for(int i=0; i<itemsToLoad; i++) {
        NSInteger index = i;
        [self downloadFileFromURL:toLoad[index] named:nil withProgress:^(float progress) {
            progresses[index] = [[NSNumber alloc] initWithFloat:progress];
            if(progressBlock) {
                float overallProgress = .0f;
                for(NSNumber *number in [progresses copy]) {
                    overallProgress += number.floatValue;
                }
                progressBlock(overallProgress/itemsToLoad);
            }
        } completion:^(NSString *filePath, NSError *error) {
            if(error) latestError = error;
            requestsOut -= 1;
            if(requestsOut <= 0) {
                if(completionBlock) {
                    completionBlock(latestError);
                }
            }
        }];
    }
}

代码解释:

所以这个方法接受一个 URL 数组。然后它检查一些文件是否已经下载并创建一个新数组,其中只包含需要下载的 URL。如果所有文件都存在或未提供 URL,则调用完成并中断操作。

接下来我创建一个可变数组并用 NSNumber 实例填充它,所有实例都具有零值。我记得将发出多少请求,并为错误创建一个占位符。我遍历所有 URL 并初始化请求,其中每个请求都将报告进度和完成情况,并且这两者都在主线程上。

所以在进度 block 中,我访问进度数组以通过索引分配新值。然后我计算平均进度并将总体进度报告给输入 block 。

请求完成会减少请求计数器的数量,当该计数器降为零时,将调用输入完成。

情况:

一切按预期工作,整个过程是正确的。给定的值都是有效的,所有文件都在服务器上并且可以访问。当应用程序没有崩溃时,一切正常。

但是当它崩溃时它崩溃了

                for(NSNumber *number in [progresses copy]) {
                    overallProgress += number.floatValue;
                }

并且崩溃是随机的,但无论如何 number.floatValue 似乎正在访问它不应该访问的内存。

我现在有一个解决方案,我将 progresses 数组替换为纯 C 指针 float *progresses = malloc(sizeof(float)*itemsToLoad); 释放完成。它似乎有效但仍然,我在这里错过了什么?带有 NSNumbers 的数组在这里不起作用的原因可能是什么?

一些附加信息:

  • 内存还可以,这是直接写入文件,即使没有,整体文件大小也相对较小
  • 磁盘空间还可以
  • 我使用的是 @(progress) 语法,但将其更改为显式分配,希望能解决这个问题
  • progresses不需要__block,我加了它以防万一
  • 在调用所有进度之前不会调用完成,即使调用了我也没有理由让应用程序崩溃

谢谢!

最佳答案

NSMutableArray 不是线程安全的。因此,即使没有明确的内存管理问题,如果 NSMutableArray 被两个不同的线程同时访问,也会发生坏事。我相信在串行队列中分派(dispatch) withProgress block 可以解决问题。

关于ios - 从数组访问 NSNumber 时崩溃,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/43865021/

有关ios - 从数组访问 NSNumber 时崩溃的更多相关文章

  1. ruby - 为什么我可以在 Ruby 中使用 Object#send 访问私有(private)/ protected 方法? - 2

    类classAprivatedeffooputs:fooendpublicdefbarputs:barendprivatedefzimputs:zimendprotecteddefdibputs:dibendendA的实例a=A.new测试a.foorescueputs:faila.barrescueputs:faila.zimrescueputs:faila.dibrescueputs:faila.gazrescueputs:fail测试输出failbarfailfailfail.发送测试[:foo,:bar,:zim,:dib,:gaz].each{|m|a.send(m)resc

  2. ruby-on-rails - 在 Ruby 中循环遍历多个数组 - 2

    我有多个ActiveRecord子类Item的实例数组,我需要根据最早的事件循环打印。在这种情况下,我需要打印付款和维护日期,如下所示:ItemAmaintenancerequiredin5daysItemBpaymentrequiredin6daysItemApaymentrequiredin7daysItemBmaintenancerequiredin8days我目前有两个查询,用于查找maintenance和payment项目(非排他性查询),并输出如下内容:paymentrequiredin...maintenancerequiredin...有什么方法可以改善上述(丑陋的)代

  3. ruby - 多次弹出/移动 ruby​​ 数组 - 2

    我的代码目前看起来像这样numbers=[1,2,3,4,5]defpop_threepop=[]3.times{pop有没有办法在一行中完成pop_three方法中的内容?我基本上想做类似numbers.slice(0,3)的事情,但要删除切片中的数组项。嗯...嗯,我想我刚刚意识到我可以试试slice! 最佳答案 是numbers.pop(3)或者numbers.shift(3)如果你想要另一边。 关于ruby-多次弹出/移动ruby​​数组,我们在StackOverflow上找到一

  4. ruby - 将数组的内容转换为 int - 2

    我需要读入一个包含数字列表的文件。此代码读取文件并将其放入二维数组中。现在我需要获取数组中所有数字的平均值,但我需要将数组的内容更改为int。有什么想法可以将to_i方法放在哪里吗?ClassTerraindefinitializefile_name@input=IO.readlines(file_name)#readinfile@size=@input[0].to_i@land=[@size]x=1whilex 最佳答案 只需将数组映射为整数:@land边注如果你想得到一条线的平均值,你可以这样做:values=@input[x]

  5. ruby - 检查 "command"的输出应该包含 NilClass 的意外崩溃 - 2

    为了将Cucumber用于命令行脚本,我按照提供的说明安装了arubagem。它在我的Gemfile中,我可以验证是否安装了正确的版本并且我已经包含了require'aruba/cucumber'在'features/env.rb'中为了确保它能正常工作,我写了以下场景:@announceScenario:Testingcucumber/arubaGivenablankslateThentheoutputfrom"ls-la"shouldcontain"drw"假设事情应该失败。它确实失败了,但失败的原因是错误的:@announceScenario:Testingcucumber/ar

  6. ruby-on-rails - 在混合/模块中覆盖模型的属性访问器 - 2

    我有一个包含模块的模型。我想在模块中覆盖模型的访问器方法。例如:classBlah这显然行不通。有什么想法可以实现吗? 最佳答案 您的代码看起来是正确的。我们正在毫无困难地使用这个确切的模式。如果我没记错的话,Rails使用#method_missing作为属性setter,因此您的模块将优先,阻止ActiveRecord的setter。如果您正在使用ActiveSupport::Concern(参见thisblogpost),那么您的实例方法需要进入一个特殊的模块:classBlah

  7. ruby - 通过 erb 模板输出 ruby​​ 数组 - 2

    我正在使用puppet为ruby​​程序提供一组常量。我需要提供一组主机名,我的程序将对其进行迭代。在我之前使用的bash脚本中,我只是将它作为一个puppet变量hosts=>"host1,host2"我将其提供给bash脚本作为HOSTS=显然这对ruby​​不太适用——我需要它的格式hosts=["host1","host2"]自从phosts和putsmy_array.inspect提供输出["host1","host2"]我希望使用其中之一。不幸的是,我终其一生都无法弄清楚如何让它发挥作用。我尝试了以下各项:我发现某处他们指出我需要在函数调用前放置“function_”……这

  8. ruby - 检查数组是否在增加 - 2

    这个问题在这里已经有了答案:Checktoseeifanarrayisalreadysorted?(8个答案)关闭9年前。我只是想知道是否有办法检查数组是否在增加?这是我的解决方案,但我正在寻找更漂亮的方法:n=-1@arr.flatten.each{|e|returnfalseife

  9. ruby - 续集在添加关联时访问many_to_many连接表 - 2

    我正在使用Sequel构建一个愿望list系统。我有一个wishlists和itemstable和一个items_wishlists连接表(该名称是续集选择的名称)。items_wishlists表还有一个用于facebookid的额外列(因此我可以存储opengraph操作),这是一个NOTNULL列。我还有Wishlist和Item具有续集many_to_many关联的模型已建立。Wishlist类也有:selectmany_to_many关联的选项设置为select:[:items.*,:items_wishlists__facebook_action_id].有没有一种方法可以

  10. Ruby Readline 在向上箭头上使控制台崩溃 - 2

    当我在Rails控制台中按向上或向左箭头时,出现此错误:irb(main):001:0>/Users/me/.rvm/gems/ruby-2.0.0-p247/gems/rb-readline-0.4.2/lib/rbreadline.rb:4269:in`blockin_rl_dispatch_subseq':invalidbytesequenceinUTF-8(ArgumentError)我使用rvm来管理我的ruby​​安装。我正在使用=>ruby-2.0.0-p247[x86_64]我使用bundle来管理我的gem,并且我有rb-readline(0.4.2)(人们推荐的最少

随机推荐