我有以下代码。我偶尔会收到SIGSEGV。我感觉我缺少有关使用块进行内存管理的东西。传递自动释放到此块的replaceUrls是否安全?修改实例变量formattedText怎么样?

    NSMutableSet* replacedUrls = [[[NSMutableSet alloc] init] autorelease];

    NSError *error = nil;
    NSDataDetector *detector = [NSDataDetector dataDetectorWithTypes:
                                (NSTextCheckingTypeLink | NSTextCheckingTypePhoneNumber)
                                                               error:&error];
    if (error) {
        return;
    }

    [detector enumerateMatchesInString:self.formattedText
              options:0
              range:NSMakeRange(0, [self.formattedText length])
              usingBlock:^(NSTextCheckingResult *result, NSMatchingFlags flags, BOOL *stop) {

            @try  {
                if (result.resultType == NSTextCheckingTypePhoneNumber) {

                    if (!result.phoneNumber) {
                        // not sure if this is possible
                        return;
                    }

                    self.formattedText = [self.formattedText stringByReplacingOccurrencesOfString:result.phoneNumber
                                                                                       withString:[NSString stringWithFormat:@"<a href=\"tel://%@\">%@</a>", result.phoneNumber, result.phoneNumber]];
                }
                else if (result.resultType == NSTextCheckingTypeLink) {

                    if (!result.URL) {
                        // not sure if this is possible
                        return;
                    }

                    NSString* fullUrl = [result.URL absoluteString];

                    if (!fullUrl) {
                        return;
                    }

                    if ([replacedUrls containsObject:fullUrl]) {
                        return;
                    }

                    // not sure if this is possible
                    if ([result.URL host] && [result.URL path]) {
                        NSString* urlWithNoScheme = [NSString stringWithFormat:@"%@%@", [result.URL host], [result.URL path]];

                        // replace all http://www.google.com to www.google.com
                        self.formattedText = [self.formattedText stringByReplacingOccurrencesOfString:fullUrl
                                                                                           withString:urlWithNoScheme];

                        // replace all www.google.com with http://www.google.com
                        NSString* replaceText = [NSString stringWithFormat:@"<a href=\"%@\">%@</a>", fullUrl, fullUrl];
                        self.formattedText = [self.formattedText stringByReplacingOccurrencesOfString:urlWithNoScheme
                                                                                           withString:replaceText];

                        [replacedUrls addObject:fullUrl];
                    }
                }
            }
            @catch (NSException* ignore) {
                // ignore any issues
            }
        }];

最佳答案

您遇到的问题似乎与内存管理有关。首先搜索字符串self.formattedText。这意味着,在进行此搜索的同时,您的NSDataDetector实例可能需要访问字符串以读取字符等。只要不释放self.formattedText,这一切都很好。通常,即使对于这样的块方法,调用者也有责任保留参数直到函数调用结束。

当在您找到匹配项的块中更改self.formattedText的值时,旧值将自动释放(假设这是retain属性)。我不知道NSDataDetector可能会进行缓存,或者与自动释放池有关的问题,等等,但是我很确定这可能会引起问题。

我的建议是您将 [NSString stringWithString:self.formattedText]传递为enumerateMatchesInString:参数,而不是普通的self.formattedText。这样,您将向NSDataDetector传递一个实例,该实例将在自动释放池耗尽后才能释放。

10-06 13:25