0

我有一个 UITableView,它从网络服务器内的 xml 文件加载内容。

  • TableView 位于标签栏控制器内;
  • [table reloadData]我在里面打电话TBXMLSuccessBlock
  • 问题是 TableView 的内容没有更新。
  • 我正在使用 TBXML 库;

单击按钮后,如何向 tableVIew 说 XML 文件已下载并显示?

- (IBAction)leftButton:(UIButton *)sender {
    if (a<=2 && a != 0) {
        a = a - 1;

        NSString *leftURL = self.getDate;
        [self loadURL:leftURL];
    }

}

- (void)loadURL:(NSString *)newURL{

    if (newURL == NULL) {
        newURL = self.getDate;
    }

    // Create a success block to be called when the asyn request completes
    TBXMLSuccessBlock successBlock = ^(TBXML *tbxmlDocument) {
        NSLog(@"PROCESSING ASYNC CALLBACK");

        [[UIApplication sharedApplication] setNetworkActivityIndicatorVisible:NO];
        // If TBXML found a root node, process element and iterate all children
        if (tbxmlDocument.rootXMLElement)
        {
            // Obtain root element
            TBXMLElement * root = tbxml.rootXMLElement;
        if (root)
        {
            _storeArray = [NSMutableArray new];
            TBXMLElement * elem_PLANT = [TBXML childElementNamed:@"news" parentElement:root];
            while (elem_PLANT !=nil)
            {
                TBXMLElement * elem_title = [TBXML childElementNamed:@"title" parentElement:elem_PLANT];
                NSString *titleName = [TBXML textForElement:elem_title];

                TBXMLElement * elem_artist = [TBXML childElementNamed:@"text" parentElement:elem_PLANT];
                NSString *artistName = [TBXML textForElement:elem_artist];

                TBXMLElement * elem_thumb = [TBXML childElementNamed:@"thumb_url" parentElement:elem_PLANT];
                NSString *thumbName = [TBXML textForElement:elem_thumb];

                NSDictionary *dictionary = [[NSDictionary alloc]initWithObjects:@[titleName, artistName, thumbName] forKeys:@[@"title", @"text", @"thumb"]];
                elem_PLANT = [TBXML nextSiblingNamed:@"news" searchFromElement:elem_PLANT];
                [_storeArray addObject:dictionary];

            }

        }

          [_tableView reloadData];
        }
    };

    // Create a failure block that gets called if something goes wrong
    TBXMLFailureBlock failureBlock = ^(TBXML *tbxmlDocument, NSError * error) {
        NSLog(@"Error! %@ %@", [error localizedDescription], [error userInfo]);
    };

    // Initialize TBXML with the URL of an XML doc. TBXML asynchronously loads and parses the file.
    tbxml = [[TBXML alloc] initWithURL:[NSURL URLWithString:newURL]
                               success:successBlock
                               failure:failureBlock];
}

编辑:按钮点击内的 ReloadData

- (IBAction)leftButton:(UIButton *)sender {
    if (a<=2 && a != 0) {
        a = a - 1;

        NSString *leftURL = self.getDate;
        [self loadURL:leftURL];
        [_tableView reloadData];
    }

    }

- (IBAction)rightButton:(UIButton *)sender {
    if (a<2) {
        a = a + 1;
        NSString *rightURL = self.getDate;
        [self loadURL:rightURL];
        [_tableView reloadData];
    }
}

应用启动时: - 加载 1 号 URL;- 显示第 1 个 URL;当我单击一次右键时: - 加载了 2 号 URL;- 什么都没发生; 当我再右击一次时: - 加载了 3 号 URL;- 显示 2 号 URL;当我左键单击一次时: - URL 2 已加载;- 显示 URL 3;当我再单击一次时: - URL 1 已加载;- 显示 URL 2;

4

2 回答 2

1

你说:

应用启动时: - 加载 1 号 URL;- 显示第 1 个 URL;当我单击一次右键时: - 加载了 2 号 URL;- 什么都没发生; 当我再右击一次时: - 加载了 3 号 URL;- 显示 2 号 URL;当我左键单击一次时: - URL 2 已加载;- 显示 URL 3;当我再单击一次时: - URL 1 已加载;- 显示 URL 2;

由此,我们知道:

  1. URL/XML 加载工作——数据在那里显示,但显示没有在它应该发生的时候发生。
  2. 表重新加载有效 - 在稍后按下按钮时,显示较早的数据,但显示不同的数据。
  3. 表重新加载发生得太早了——这就是它显示先前 URL/XML 数据的原因。

您的按钮操作正在(间接)使用带有成功/失败块的库进行 URL 加载调用,因此大致如下:

  1. loadUrl:运行 TBXML 的IBAction 调用initWithURL:success:failure:
  2. initWithURL:success:failure:存储成功和失败块并在单独的线程(“在后台”)上启动实际的网络操作,然后返回,几乎可以肯定在网络操作完成之前。
  3. IBAction 调用reloadData.
  4. 稍后,在后台,网络操作完成并运行成功或失败块。

因此,将reloadData调用移动到成功块中,以便在处理 XML 数据后重新加载表。

但最初这崩溃了一个错误,其中包括:

这可能是从辅助线程调用 UIKit 的结果。

UIKit 调用——粗略地说,任何影响用户界面的方法调用——都必须在主线程上进行。对于一些使用块回调的异步库(特别是 AFNetworking),正是因为这个原因,块回调才能保证在主线程上运行。显然,对于 TBXML,块回调在后台运行,可能在执行实际网络加载操作的同一线程上。所以,我们需要一些方法来reloadData调用UITableView主线程上的发生。

有几种方法可以做到这一点。我的首选通常是使用主操作队列(我们使用mainQueue类方法 on获得NSOperationQueue),这是在主线程上执行的操作队列:

[[NSOperationQueue mainQueue] addOperationWithBlock:^{
    [_tableView reloadData];
}];

然后,主操作队列会尽快在主线程上自动处理执行该块。或者,但仍处于 Objective-C/Cocoa/CocoaTouch 级别,您可以performSelectorOnMainThread:withObject:waitUntilDone:在表视图上使用:

[_tableView performSelectorOnMainThread:@selector(reloadData)
                             withObject:nil
                          waitUntilDone:NO];

(或者YES等到完成——在这种情况下可能无关紧要),这具有完全相同的效果。第三种选择,虽然在这种情况下通常不是首选,但直接使用 Grand Central Dispatch:

dispatch_async(dispatch_get_main_queue(), ^{
    [_tableView reloadData];
}];

(但是当有更高级别的选项时,如NSOperationQueueroute 或performSelectorOnMainThread:withObject:waitUntilDone:route,应该使用它们来代替这种较低级别的 C Grand Central Dispatch 例程;另外,更改dispatch_async为与前面示例中更改为dispatch_sync的效果相同) .waitUntilDone:NOwaitUntilDone:YES

于 2013-03-23T00:11:29.953 回答
-1

正如你所说,你有调用方法 reloadData 然后它应该刷新表格视图..我认为问题不存在,你是否尝试过不使用标签栏控制器的人..有时可能会出现标签栏控制器问题。 ..所以只需尝试一次..&让我们知道..或者我们以其他方式帮助你

于 2013-03-23T04:40:56.847 回答