与NSXMLParser和ARC的EXC_BAD_ACCESS

时间:2012-12-19 15:37:50

标签: ios automatic-ref-counting exc-bad-access nsxmlparser

在iOS开发中初学者,我将尝试实现一个通过XML文件显示RSS提要的应用程序。

在我的viewDidLoad课程的UITableView中,我使用UIActivityIndicator等待数据加载。

但是,在应用程序返回主线程的那一刻,我在EXC_BAC_ACCESS函数的末尾有一个parseXMLStart代码2。我不明白为什么......

这里出现错误信息:

Thread 6 : 0-[NSXMLParser dealloc]
Message  : EXC_BAC_ACCESS (code=2, address=0xc)
Line     : 0xbb0840:  movl   (%eax,%ecx), %ecx

我不知道我的错误是什么以及在哪里。我该如何解决?


这是我的代码:

=> Class Type :: UITableViewController    

>> Header

@interface DataListViewController : UITableViewController {
    UIActivityIndicatorView *activityView;
    NSMutableArray *dataFromXML;
}

- (void)parseXMLStart;
- (void)parseXMLDone;

@end

>> Main

@implementation DataListViewController

- (void)viewDidLoad
{
    [super viewDidLoad];

    self.view.backgroundColor = [UIColor whiteColor];
    self.title = @"View 1";

    activityView = [[UIActivityIndicatorView alloc] initWithActivityIndicatorStyle:UIActivityIndicatorViewStyleGray];
    activityView.center = self.view.center;

    [self performSelectorInBackground:@selector(parseXMLStart) withObject:nil];

    [activityView startAnimating];
    [activityView setHidesWhenStopped:YES];

    [self.view addSubview:activityView];
}

#pragma mark - UIActivityIndicator Methods

- (void)parseXMLStart
{
    // To Show the animation
    sleep(1);

    dataFromXML = [[NSMutableArray alloc] init];

    // COMMENT TO TEST /*

    [dataFromXML addObject:@"Element 1"];
    [dataFromXML addObject:@"Element 2"];
    [dataFromXML addObject:@"Element 3"];

    // */ COMMENT TO TEST

    // ------------------------------------------------------------------------------------------------------------------------------------

    // UNCOMMENT TO TEST
    /*

    NSString *filePath = [[NSBundle mainBundle] pathForResource:@"words" ofType:@"xml"];
    NSURL *url = [[NSURL alloc] initWithString:[[NSString stringWithFormat:@"file://%@",filePath] stringByAddingPercentEscapesUsingEncoding:NSASCIIStringEncoding]];

     XML2ObjectParser *parserWords = [[XML2ObjectParser alloc] parseXMLAtURL:url toObject:@"Word" parseError:nil];

    NSLog(@">> parserWords Items Count :: %i", parserWords.items.count);

    for (int i = 0; i < [parserWords.items count]-1; i++) {
        Word *aWord = [[Word alloc] init];
        aWord = (Word *)[[parserWords items] objectAtIndex:i];
        [dataFromXML addObject:aWord];
    }

    NSLog(@">> dataFromXML Count :: %i", dataFromXML.count);

    */
    // UNCOMMENT TO TEST

    // --------------------------------------------------------------------------------------------------------------------------------------------

    // EXC_BAD_ACCESS (code=2, address=0xc)
    // Thread 6 : 0-[NSXMLParser dealloc]
    // 0xbad840:  movl   (%eax,%ecx), %ecx

    // --------------------------------------------------------------------------------------------------------------------------------------------

    [self performSelectorOnMainThread:@selector(parseXMLDone) withObject:nil waitUntilDone:YES];
}

- (void)parseXMLDone
{
    [activityView stopAnimating];
    [self.tableView reloadData];
}

# pragma mark - Table View Method

- (NSInteger)numberOfSectionsInTableView:(UITableView *)tableView
{
    return 1;
}

- (NSInteger)tableView:(UITableView *)tableView numberOfRowsInSection:(NSInteger)section
{
    return [dataFromXML count];
}

- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath
{
    static NSString *cellID = @"CellID";
    UITableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:cellID];

    if (cell == nil) {
        cell = [[UITableViewCell alloc] initWithStyle:UITableViewCellStyleDefault reuseIdentifier:cellID];
    }

    cell.textLabel.text = [dataFromXML objectAtIndex:[indexPath row]];

    return cell;
}

- (void)tableView:(UITableView *)tableView didSelectRowAtIndexPath:(NSIndexPath *)indexPath
{
    DataListDetailViewController *_dataListDetailViewController = [[DataListDetailViewController alloc] init];

    _dataListDetailViewController.title = [dataFromXML objectAtIndex:[indexPath row]];

    [self.navigationController pushViewController:_dataListDetailViewController animated:YES];
}

@end

2 个答案:

答案 0 :(得分:1)

在我看来,您的选择器输入错误。你有@selector(parsingXMLDone),它应该是@selector(parseXMLDone)

答案 1 :(得分:1)

我发现了错误。

这不是我的错,而是ARC。

我解释说,在解析XML时,它已经完成,后台线程被终止,并且在ARC想要解除分配NSXMLParser之后。

这就是为什么它会导致EXC_BAD_ACCESS,因为ARC想要释放已经解除分配的对象(AKA NSXMLParser)。

解决方案是在没有ARC的情况下编译我的类,在目标的构建阶段使用标志“fno-objc-arc”。

请参阅:NSXMLParser gives EXC_BAD_ACCESS only with ARC enabled

感谢您的帮助。