簡體   English   中英

TBXML tableview內容不更新

[英]TBXML tableview contents don`t update

我有一個UITableView,它從Web服務器內的xml文件加載內容。

  • TableView在標簽欄控制器內;
  • 我在TBXMLSuccessBlock內調用[table reloadData] ;
  • 問題是TableView的內容沒有更新。
  • 我正在使用TBXML庫;

單擊“按鈕”后,如何說出表看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];
}

編輯:在buttonClick內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];
    }
}

應用啟動時:-加載了URL號1; -顯示URL號1; 當我右鍵單擊一次時:-URL號2已加載; - 什么都沒發生; 當我再次單擊一次時:-URL號3已加載; -顯示URL號2; 當我單擊左鍵一次時:-URL 2已加載; -顯示網址3; 當我再點擊一次時:-URL 1已加載; -顯示網址2;

你說:

應用啟動時:-加載了URL號1; -顯示URL號1; 當我右鍵單擊一次時:-URL號2已加載; - 什么都沒發生; 當我再次單擊一次時:-URL號3已加載; -顯示URL號2; 當我單擊左鍵一次時:-URL 2已加載; -顯示網址3; 當我再點擊一次時:-URL 1已加載; -顯示網址2;

由此,我們知道:

  1. URL / XML加載有效-可以顯示數據,但是在適當的時候不會顯示。
  2. 重新加載表的工作原理—在稍后按下按鈕時,將顯示較早的數據,但是將顯示不同的數據。
  3. 表重新加載還為時過早,這就是為什么它顯示以前的URL / XML數據的原因。

您的按鈕操作是(間接地)使用帶有成功/失敗塊的庫進行URL加載調用,因此大致是這樣的:

  1. IBAction調用loadUrl:它運行TBXML的initWithURL:success:failure:
  2. initWithURL:success:failure:存儲成功和失敗塊,並在單獨的線程(“在后台”)中啟動實際的網絡操作,並幾乎可以肯定在網絡操作完成之前返回。
  3. IBAction調用reloadData
  4. 稍后,在后台,網絡操作完成,並且運行成功或失敗塊。

因此,將reloadData調用移到成功塊中可以使它在處理XML數據之后重新加載表。

但是最初這崩潰了,並包含以下錯誤:

這可能是從輔助線程調用UIKit的結果。

UIKit調用(大概是影響用戶界面的任何方法調用)都必須在主線程上進行。 對於某些使用塊回調的異步庫(尤其是AFNetworking),由於這個原因,可以保證塊回調在主線程上運行。 顯然,使用TBXML,塊回調在后台運行,可能在執行實際網絡加載操作的同一線程上。 因此,我們需要某種方法使對UITableViewreloadData調用發生在主線程上。

有幾種方法可以做到這一點。 我的第一個偏好通常是使用主操作隊列(使用mainQueue上的mainQueue類方法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];
}];

(但是,當有更高級別的選項(例如NSOperationQueue路由或performSelectorOnMainThread:withObject:waitUntilDone:路由)時,應使用它們代替此類低級別的C Grand Central Dispatch例程;而且,將dispatch_async更改為dispatch_sync具有相同的功能效果就像在上一個示例waitUntilDone:NO更改為waitUntilDone:YES )。

就像你說的那樣,你有調用方法reloadData然后它應該刷新表視圖。.我認為那是不存在的問題,你是否嘗試過不使用標簽欄控制器的人。。 ..所以只需嘗試一次..&讓我們知道..或者我們以其他方式幫助您

暫無
暫無

聲明:本站的技術帖子網頁,遵循CC BY-SA 4.0協議,如果您需要轉載,請注明本站網址或者原文地址。任何問題請咨詢:yoyou2525@163.com.

 
粵ICP備18138465號  © 2020-2024 STACKOOM.COM