代码之家  ›  专栏  ›  技术社区  ›  mensch

刷新XML数据并更新UITableView

  •  0
  • mensch  · 技术社区  · 15 年前

    我正在开发一个iPhone应用程序,它使用TableView来显示XML数据。XML数据来自外部资源,并被解析并放入对象中,以用作表数据源。应用程序使用一个UItaBar和多个视图控制器,所有这些控件都是以编程方式创建并使用相同的数据源。

    一切都很好,很漂亮,但是我想实现一个刷新按钮,这样用户就可以刷新内容(全局,所以所有的视图控制器都应该更新)。解析器将再次查询XML并创建一个新对象。我遇到的问题是,我似乎无法用新数据重新填充TableView。我得到更新的数据对象。但是,实际上更新TableView是一个问题。我已经试过了 setDataSource 并打电话 reloadData ,但这会导致由于未识别的选择器而崩溃。

    XML资料是从我的AppDelegate调用的,所有的分析逻辑都在parser.m中。刷新函数在rootviewcontroller.m中调用,它实现了uitableviewdatasource协议:

    - (void)refreshXMLFeed:(id)sender {
      NSArray *tableControllersData = [appDelegate getData];
      [self.tableView setDataSource: tableControllersData];
      [self.tableView reloadData];  
    }
    

    我该如何处理这个问题?应该获取新数据并像我一直试图完成的那样在rootviewcontroller中重新加载表视图。或者应该在appdelegate中触发数据分析,并且只在rootviewcontroller中重新加载TableView。

    如果需要,我可以用必要的代码更新我的问题,我不确定在这一点上哪些部分是相关的。

    rootview控制器.h:

    #import <UIKit/UIKit.h>
    
    @interface RootViewController : UITableViewController {
      MyAppDelegate *appDelegate;
      NSArray *tableDataArray;
    }
    
    @property (nonatomic, retain) NSArray *tableDataArray;
    
    - (IBAction)refreshXMLFeed:(id)sender;
    - (id)initWithNibName:(NSString *)nibNameOrNil bundle:(NSBundle *)nibBundleOrNil tableDataSource:(NSArray*)tableData;
    
    @end
    

    rootview控制器.m:

    #import "CustomCell.h"
    #import "MyAppDelegate.h"
    #import "RootViewController.h"
    #import "DetailViewController.h"
    
    @implementation RootViewController
    @synthesize tableDataArray;
    
    - (void)viewDidLoad {
      [super viewDidLoad];
    }
    
    //Override the default initWithNibName method
    - (id)initWithNibName:(NSString *)nibNameOrNil bundle:(NSBundle *)nibBundleOrNil tableDataSource:(NSArray*)tableData {
      if (self = [super initWithNibName:nibNameOrNil bundle:nibBundleOrNil]) {
        // Custom initialization
        tableDataArray = [tableData retain];   
      }
      return self;
    }
    
    -(void)viewWillAppear:(BOOL)animated {
      appDelegate = (MyAppDelegate *)[[UIApplication sharedApplication] delegate];    
      [super viewWillAppear:animated];
      //Set the colour of the navigationController and add buttons
      self.navigationController.navigationBar.tintColor = [UIColor blackColor];
    
      //Add the refresh button
      UIBarButtonItem* refreshButton = [[UIBarButtonItem alloc] initWithBarButtonSystemItem:UIBarButtonSystemItemRefresh target:self action:@selector(refreshXMLFeed:)];
      [self.navigationItem setLeftBarButtonItem:refreshButton animated:YES];
      [refreshButton release];  
    }
    
    #pragma mark Table
    
    - (NSInteger)tableView:(UITableView *)tableView numberOfRowsInSection:(NSInteger)section {
      return [self.tableDataArray count];
    }
    
    - (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath {
      static NSString *CustomCellIdentifier = @"CustomCellIdentifier";
      CustomCell *cell = (CustomCell *)[tableView dequeueReusableCellWithIdentifier: CustomCellIdentifier];
      if (cell == nil) {
        NSArray *cellNib = [[NSBundle mainBundle] loadNibNamed:@"CustomCell" owner:self options:nil];
        for (id oneObject in cellNib) {
          if ([oneObject isKindOfClass:[CustomCell class]]) {
            cell = (CustomCell *)oneObject;
          }
        }
      }
    
      NSUInteger row = [indexPath row];
      NSDictionary *rowData = [self.tableDataArray objectAtIndex:row];
      cell.colorLabel.text = [rowData objectForKey:@"Type"];
      cell.nameLabel.text = [rowData objectForKey:@"Name"];
      UIImage *image = [UIImage imageNamed:[rowData objectForKey:@"Icon"]];
      cell.image = image;
      return cell;
    }
    
    - (void)tableView:(UITableView *)tableView didSelectRowAtIndexPath:(NSIndexPath *)indexPath {
      NSString *selectedRow = [tableDataArray objectAtIndex:indexPath.row];
    
      DetailViewController *detailViewController = [[DetailViewController alloc] initWithNibName:@"DetailView" bundle:[NSBundle mainBundle]];
      detailViewController. selectedRow = selectedRow;
      [self.navigationController pushViewController:detailViewController animated:YES];
    
      UIBarButtonItem *backButton = [[UIBarButtonItem alloc] init];
      backButton.title = @"Back";
      self.navigationItem.backBarButtonItem = backButton;
      [backButton release];  
      [detailViewController release];
      detailViewController = nil;
    }
    
    #pragma mark Refresh
    
    - (void)refreshXMLFeed:(id)sender {
      NSArray *tableControllersData = [appDelegate getData];
      [self.tableView setDataSource: tableControllersData];
      [self.tableView reloadData];  
    }
    
    - (void)didReceiveMemoryWarning {
        // Releases the view if it doesn't have a superview.
        [super didReceiveMemoryWarning];
    
        // Release any cached data, images, etc that aren't in use.
    }
    
    - (void)viewDidUnload {
        // Release any retained subviews of the main view.
        // e.g. self.myOutlet = nil;
    }
    
    
    - (void)dealloc { 
        [super dealloc];
    }
    
    @end
    
    2 回复  |  直到 13 年前
        1
  •  3
  •   Matt Long    15 年前

    您应该只设置一次数据源,而不应该是NSarray。从中提取记录的数据容器可以更改,但数据源应始终相同。在大多数情况下,数据源应该只是视图控制器。然后视图控制器应该在 UITableViewDataSource protocol 包括:

    – tableView:cellForRowAtIndexPath:  required method
    – numberOfSectionsInTableView:
    – tableView:numberOfRowsInSection:  required method
    – sectionIndexTitlesForTableView:
    – tableView:sectionForSectionIndexTitle:atIndex:
    – tableView:titleForHeaderInSection:
    – tableView:titleForFooterInSection:
    

    NSarray不响应这些方法中的任何一个,这就是为什么您收到无法识别的选择器消息的原因。你必须自己实现它们。

    你应该熟悉 Table View Programming Guide 了解如何有效地使用表视图。

    最好的问候。

    更新:这里有一些代码可以帮助您。在rootviewcontroller中,用-viewdidload中的alloc/init实例化nsarray。称之为项目:

    - (void)viewDidLoad;
    {
        [super viewDidLoad];
        items = [[NSArray alloc] init];
    }
    

    然后像这样实现表视图的数据源委托:

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

    然后需要实现cellForRowatindexPath:

    - (UITableViewCell *)tableView:(UITableView *)tv cellForRowAtIndexPath:(NSIndexPath *)indexPath
    {
        UITableViewCell *cell;
    
        cell = [tv dequeueReusableCellWithIdentifier:@"Cell"];
        if (cell == nil)
        {
            cell = [[[UITableViewCell alloc] initWithFrame:CGRectZero reuseIdentifier:@"Cell"] autorelease];
    
        }
    
        // Get your item from the items array here
        NSDictionary *item = [items objectAtIndex:[indexPath row]];
        NSString *text = [item valueForKey:@"text"];
    
        [[cell textLabel] setText:text];
    
        return cell;
    
    }
    

    此代码假定NSarray中的对象是NSDictionary。如果正在使用某个自定义对象,请将该索引路径处的对象强制转换为自定义类型,然后使用它的字段。

    当有新数据可用并且需要重新加载表视图时,只需重新分配项nsarray,然后在表视图上调用reload data。假设您有这样的重新加载:

    - (void)didReciveNewData:(NSArray*)newItems;
    {
        if (items) [items release], items = nil;
        items = [newItems copy];
        [tableView reloadData];
    }
    

    这将导致表视图查询其视图控制器,以获取要显示的行数以及通过访问NSARRAY的计数和内容而再次提供的每行的单元格。

    Hth.

        2
  •  0
  •   mml    15 年前

    我猜您的数据是在getdata方法中异步加载的(如果您不是,您应该是),它在用户界面最不期望的时候更新/使对数据源的引用无效,当用户界面试图用指向其数据的过时指针呈现数据时,会导致崩溃。

    只需确保从不从其他线程修改TableControllersData的内容。

    或者你想使用coredata,在这种情况下,我不知道。