#ios
#iOS
Вопрос:
Я пытаюсь освоить разработку для IOS, я создал проект из примера приложения RSS-канала, которое загружает данные из файла XML-канала на мой сервер. Это работает нормально, но я хочу, чтобы оно обновлялось при нажатии кнопки home (многозадачность ios). Я попробовал [table reloaddata] и поместил его во все разделы viewdidload / viewdiddisappear, но он не работает и не останавливается на точках останова, которые я установил для них всех.
@implementation RootViewController
- (void)viewDidLoad {
// Add the following line if you want the list to be editable
// self.navigationItem.leftBarButtonItem = self.editButtonItem;
//[newsTable reloaddata];
}
- (NSInteger)numberOfSectionsInTableView:(UITableView *)tableView {
return 1;
}
- (NSInteger)tableView:(UITableView *)tableView numberOfRowsInSection:(NSInteger)section {
return [stories count];
}
- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath {
static NSString *MyIdentifier = @"MyIdentifier";
UITableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:MyIdentifier];
if (cell == nil) {
cell = [[[UITableViewCell alloc] initWithFrame:CGRectZero reuseIdentifier:MyIdentifier] autorelease];
}
// Set up the cell
int storyIndex = [indexPath indexAtPosition: [indexPath length] - 1];
[cell setText:[[stories objectAtIndex: storyIndex] objectForKey: @"title"]];
return cell;
}
- (void)tableView:(UITableView *)tableView didSelectRowAtIndexPath:(NSIndexPath *)indexPath {
// Navigation logic
int storyIndex = [indexPath indexAtPosition: [indexPath length] - 1];
NSString * storyLink = [[stories objectAtIndex: storyIndex] objectForKey: @"link"];
// clean up the link - get rid of spaces, returns, and tabs...
storyLink = [storyLink stringByReplacingOccurrencesOfString:@" " withString:@""];
storyLink = [storyLink stringByReplacingOccurrencesOfString:@"n" withString:@""];
storyLink = [storyLink stringByReplacingOccurrencesOfString:@" " withString:@""];
NSLog(@"link: %@", storyLink);
// open in Safari
[[UIApplication sharedApplication] openURL:[NSURL URLWithString:storyLink]];
}
- (void)viewWillAppear:(BOOL)animated {
[super viewWillAppear:animated];
[newsTable reloadData];
}
- (void)viewDidAppear:(BOOL)animated {
[super viewDidAppear:animated];
if ([stories count] == 0) {
[newsTable reloadData];
NSString * path = @"http://www.myserver.co.uk/test.xml";
[self parseXMLFileAtURL:path];
//[NSTimer scheduledTimerWithTimeInterval:10 target:self selector:@selector(reloadData) userInfo:nil repeats:FALSE];
}
cellSize = CGSizeMake([newsTable bounds].size.width, 60);
}
-(void)viewWillDisappear:(BOOL)animated {
[newsTable reloadData];
}
- (void)viewDidDisappear:(BOOL)animated {
[newsTable reloadData];}
- (void)applicationDidBecomeActive:(UIApplication *)application {
/*
Restart any tasks that were paused (or not yet started) while the application was inactive. If the application was previously in the background, optionally refresh the user interface.
*/
[newsTable reloadData];
}
- (void)applicationWillEnterForeground:(UIApplication *)application {
/*
Called as part of transition from the background to the inactive state: here you can undo many of the changes made on entering the background.
*/
[newsTable reloadData];
}
- (void)parserDidStartDocument:(NSXMLParser *)parser{
NSLog(@"found file and started parsing");
}
- (void)parseXMLFileAtURL:(NSString *)URL
{
stories = [[NSMutableArray alloc] init];
//you must then convert the path to a proper NSURL or it won't work
NSURL *xmlURL = [NSURL URLWithString:URL];
// here, for some reason you have to use NSClassFromString when trying to alloc NSXMLParser, otherwise you will get an object not found error
// this may be necessary only for the toolchain
rssParser = [[NSXMLParser alloc] initWithContentsOfURL:xmlURL];
// Set self as the delegate of the parser so that it will receive the parser delegate methods callbacks.
[rssParser setDelegate:self];
// Depending on the XML document you're parsing, you may want to enable these features of NSXMLParser.
[rssParser setShouldProcessNamespaces:NO];
[rssParser setShouldReportNamespacePrefixes:NO];
[rssParser setShouldResolveExternalEntities:NO];
[rssParser parse];
}
- (void)parser:(NSXMLParser *)parser parseErrorOccurred:(NSError *)parseError {
NSString * errorString = [NSString stringWithFormat:@"Unable to download story feed from web site (Error code %i )", [parseError code]];
NSLog(@"error parsing XML: %@", errorString);
UIAlertView * errorAlert = [[UIAlertView alloc] initWithTitle:@"Error loading content" message:errorString delegate:self cancelButtonTitle:@"OK" otherButtonTitles:nil];
[errorAlert show];
}
- (void)parser:(NSXMLParser *)parser didStartElement:(NSString *)elementName namespaceURI:(NSString *)namespaceURI qualifiedName:(NSString *)qName attributes:(NSDictionary *)attributeDict{
//NSLog(@"found this element: %@", elementName);
currentElement = [elementName copy];
if ([elementName isEqualToString:@"item"]) {
// clear out our story item caches...
item = [[NSMutableDictionary alloc] init];
currentTitle = [[NSMutableString alloc] init];
currentDate = [[NSMutableString alloc] init];
currentSummary = [[NSMutableString alloc] init];
currentLink = [[NSMutableString alloc] init];
}
}
- (void)parser:(NSXMLParser *)parser didEndElement:(NSString *)elementName namespaceURI:(NSString *)namespaceURI qualifiedName:(NSString *)qName{
//NSLog(@"ended element: %@", elementName);
if ([elementName isEqualToString:@"item"]) {
// save values to an item, then store that item into the array...
[item setObject:currentTitle forKey:@"title"];
[item setObject:currentLink forKey:@"link"];
[item setObject:currentSummary forKey:@"summary"];
[item setObject:currentDate forKey:@"date"];
[stories addObject:[item copy]];
NSLog(@"adding story: %@", currentTitle);
}
}
- (void)parser:(NSXMLParser *)parser foundCharacters:(NSString *)string{
//NSLog(@"found characters: %@", string);
// save the characters for the current item...
if ([currentElement isEqualToString:@"title"]) {
[currentTitle appendString:string];
} else if ([currentElement isEqualToString:@"link"]) {
[currentLink appendString:string];
} else if ([currentElement isEqualToString:@"description"]) {
[currentSummary appendString:string];
} else if ([currentElement isEqualToString:@"pubDate"]) {
[currentDate appendString:string];
}
}
- (void)parserDidEndDocument:(NSXMLParser *)parser {
[activityIndicator stopAnimating];
[activityIndicator removeFromSuperview];
NSLog(@"all done!");
NSLog(@"stories array has %d items", [stories count]);
[newsTable reloadData];
}
- (BOOL)shouldAutorotateToInterfaceOrientation:(UIInterfaceOrientation)interfaceOrientation {
// Return YES for supported orientations
return (interfaceOrientation == UIInterfaceOrientationPortrait);
}
- (void)didReceiveMemoryWarning {
[super didReceiveMemoryWarning]; // Releases the view if it doesn't have a superview
// Release anything that's not essential, such as cached data
}
- (void)dealloc {
[currentElement release];
[rssParser release];
[stories release];
[item release];
[currentTitle release];
[currentDate release];
[currentSummary release];
[currentLink release];
[super dealloc];
}
@end
Ответ №1:
Мне нравится использовать уведомления для решения этой ситуации. Добавьте эту строку кода в свой метод ‘viewDidLoad’
[[NSNotificationCenter defaultCenter] addObserver:self
selector:@selector(updateRSSFeed)
name:UIApplicationWillEnterForegroundNotification
object:nil];
Метод ‘updateRSSFeed’ сделает что-то вроде этого
- (void) updateRSSFeed
{
NSLog( @"Feed Me!!!");
// Code to restart loading the data from the RSS feed and ultimately reloading
// the table view.
}
И не забудьте удалить observer, поэтому добавьте это в метод ‘dealloc’
[[NSNotificationCenter defaultCenter] removeObserver: self];
В сложном приложении с несколькими контроллерами представления контроллер представления может быть освобожден без завершения работы приложения, а уведомление может быть отправлено несуществующему экземпляру, и вы получите сбой.
Уведомления очень полезны, поэтому изучите документацию и другие источники, чтобы выяснить, как их использовать.
Наконец, методы ‘applicationDidBecomeActive:’ и ‘applicationWillEnterForeground:’ никогда не будут вызываться в корневом контроллере представления, поскольку они являются частью протокола делегирования приложений, к которому привязан ваш AppDelegate.
Ответ №2:
Эй, если вы хотите реагировать на переход приложения в фоновый режим, вам следует реализовать
- (void)applicationDidEnterBackground:(UIApplication *)application
в вашем AppDelegate и запустите перезагрузку данных оттуда. viewWillAppear не вызывается, когда приложение переходит в фоновый режим (т. Е. Когда вы «сворачиваете» его, нажимаете кнопку home или переключаетесь на другое приложение.
Приветствия
Ответ №3:
Я не совсем понимаю, откуда вы звоните [tableView reloadData]
. Судя по вашему описанию, вы должны вызывать его во время applicationWiilResignActive:
, но это, скорее всего, не запустится, если вы не укажете ОС, которую хотите загрузить в фоновом режиме.
Однако я полагаю, что вы должны уже начать эту загрузку до вызова этого метода, что объяснит, почему вы не видите ни перезагрузки таблицы, ни вызываемых точек останова.
Возможно, я неправильно понял, как вы выполняете вызов, но я думаю, что у вас фундаментальная проблема с дизайном, поскольку вы не можете запустить эту перезагрузку в момент деактивации.