Приращение `static int` вызывает SIGSEGV SEGV_ACCERR - PullRequest
1 голос
/ 30 мая 2011

Я отлаживаю сбой, о котором сообщается как:

Exception Type:  SIGSEGV
Exception Codes: SEGV_ACCERR

Сбой происходит на линии, которая numberOfFails++.

Приложение использует ASIHTTP. Я лично очень предпочитаю использовать NSURLConnection. Я бы никогда не повторил автоматически запрос на NSURLConnection, если бы он потерпел неудачу, потому что я никогда не видел, чтобы он терпел неудачу, когда это не должно происходить. Я бы предпочел просто дать пользовательскому интерфейсу кнопку обновления или показать UIAlertView с кнопкой, чтобы повторить попытку, или что-то в этом роде.

В любом случае, чтобы сотрудничать с другими членами команды, я пытаюсь решить эту проблему, не заменяя сейчас ASIHTTP на NSURLConnection.

Запрос запускается с чем-то вроде:

- (void)getResources:(CLLocation *)location withQuery:(NSString *)query {
    NSURL *url = [NSURL URLWithString:[NSString stringWithString:@"https://example.com/"]];
    self.resourcesAPIRequest = [ASIFormDataRequest requestWithURL:url];
    [resourcesAPIRequest setPostValue:[Model instance].oauth_token forKey:@"oauth_token"];
    [resourcesAPIRequest setPostValue:[[NSNumber numberWithDouble:location.coordinate.latitude] stringValue] forKey:@"latitude"];
    [resourcesAPIRequest setPostValue:[[NSNumber numberWithDouble:location.coordinate.longitude] stringValue] forKey:@"longitude"];
    [resourcesAPIRequest setPostValue:query forKey:@"query"];
    [resourcesAPIRequest setDelegate:self];
    [resourcesAPIRequest setDidFinishSelector:@selector(resourcesAPIReturned:)];
    resourcesAPIRequest.userInfo = [NSDictionary dictionaryWithObjectsAndKeys:NSStringFromSelector(@selector(getResources:withQuery:)), @"repeatSelector", location, @"argument1", query, @"argument2", nil];   
    [resourcesAPIRequest startAsynchronous];
}

Я заметил одну вещь: : <ASIHTTPRequestDelegate> отсутствует в заголовочном файле, но этот метод обратного вызова все еще вызывается OK:

#define maximumNumberOfFails 50

- (void)requestFailed:(ASIFormDataRequest *)request {
    static int numberOfFails = 0;

    if (numberOfFails < maximumNumberOfFails) {
        [NSThread sleepForTimeInterval:sleepTimeInSeconds];
        if ([request.userInfo objectForKey:@"argument2"]) {
            [self performSelector:NSSelectorFromString([request.userInfo objectForKey:@"repeatSelector"]) withObject:[request.userInfo objectForKey:@"argument1"] withObject:[request.userInfo objectForKey:@"argument2"]];
        } else if ([request.userInfo objectForKey:@"argument1"]) {
            [self performSelector:NSSelectorFromString([request.userInfo objectForKey:@"repeatSelector"]) withObject:[request.userInfo objectForKey:@"argument1"]];
        } else {
            [self performSelector:NSSelectorFromString([request.userInfo objectForKey:@"repeatSelector"])];
        }
        numberOfFails++;
    } else {
        UIAlertView *alert = [[UIAlertView alloc] initWithTitle:@"Problem" message:@"There was a problem connecting to the servers.  Please make sure you have an Internet connection." delegate:nil cancelButtonTitle:@"Ok" otherButtonTitles:nil];
        [alert show];
        [alert release];
        numberOfFails = 0;
    }
}

Кроме того, я думаю, что static int numberOfFails должно быть static NSUInteger numberOfFails. И я заметил, что запрос запускается с startAsynchronous. static int numberOfFails атомный? Возможно, поэтому мы получаем ошибку SEGV_ACCERR (недопустимые разрешения для сопоставленного объекта).

Мысли

1 Ответ

3 голосов
/ 30 мая 2011

Проблема, вероятно, не имеет ничего общего с вашей статической переменной.

Выполняется ли requestFailed: в основном потоке или в фоновом потоке?

Если это фоновый поток, вам нужно будет использовать performSelectorOnMainThread:withObject:.

Если он находится в главном потоке, вам может потребоваться пройти через цикл выполнения перед выполнением нового HTTP-запроса.Для этого используйте performSelector:withObject:afterDelay: и передайте '0.0' в качестве задержки.

Вы заметите, что оба этих метода допускают только один параметр метода.Как правило, вы передаете свои параметры в NSDictionary, а не пытаетесь предварительно проанализировать количество параметров, как вы это делаете.

...