在导航控制器中按下后退按钮时,是否可以不取消分配视图控制器?

时间:2013-07-31 05:06:56

标签: iphone ios objective-c dealloc detailview

这是我的问题:我有一堆带有一堆单元格的tableview。 Core Data使用NSFetchedResultsController将Task对象加载到单元格中。现在我有它所以每个单元格都有一个DetailViewController,它存储在一个字典中,如下所示:

-(void)tableView:(UITableView *)tableView didSelectRowAtIndexPath:(NSIndexPath *)indexPath{
    DetailViewController *detailVC;
    if (![self.detailViewsDictionary.allKeys containsObject:@(indexPath.row)]){
        detailVC = [[DetailViewController alloc]initWithNibName:@"DetailViewController" bundle:nil];
        [self.detailViewsDictionary setObject:detailVC forKey:@(indexPath.row)];
    }else{
        detailVC = self.detailViewsDictionary[@(indexPath.row)];
    }
        Tasks *task = [[self fetchedResultsController] objectAtIndexPath:indexPath];
        detailVC.testTask = task;
        [[self navigationController] pushViewController:detailVC animated:YES];
}

每个DetailViewController都有一个timer属性,它从Task对象获取其时间间隔,该对象与tableview中任何给定索引路径的单元格相关。这是我的详细视图控制器的代码:

- (void)viewDidLoad {
    [super viewDidLoad];
    [NSThread detachNewThreadSelector:@selector(startTimer:) toTarget:self withObject:nil];
}
-(IBAction)startTimer:(id)sender{
//default value of showButtonValue when first loaded is 1
    if (testTask.showButtonValue == 1) {
        [startButton setTitle:@"Start" forState:UIControlStateNormal];
        timer = [NSTimer scheduledTimerWithTimeInterval:1.0 target:self selector:@selector(timerAction:) userInfo:nil repeats:YES];
        testTask.showButtonValue = 3;
    } else if (testTask.showButtonValue == 2) {
        [startButton setTitle:@"Resume" forState:UIControlStateNormal];
        [timer invalidate];
        timer = nil;
        testTask.showButtonValue = 3;
    } else if (testTask.showButtonValue == 3){
        [startButton setTitle:@"Pause" forState:UIControlStateNormal];
        timer = [NSTimer scheduledTimerWithTimeInterval:1.0 target:self selector:@selector(timerAction:) userInfo:nil repeats:YES];
        testTask.showButtonValue = 2;
    }
}
-(void)viewWillAppear:(BOOL)animated{
    [super viewWillAppear:animated];
    nameTextField.text = testTask.taskName;
    [timerLabel setFont:[UIFont fontWithName:@"Arial" size:60]];
    NSUInteger seconds = (NSUInteger)round(testTask.timeInterval);
    NSString *string = [NSString stringWithFormat:@"%02u:%02u:%02u",
                        seconds / 3600, (seconds / 60) % 60, seconds % 60];
    timerLabel.text = string;
    [[self navigationItem] setTitle:[testTask taskName]];

    [timerLabel setNeedsDisplay];
    [self.view setNeedsDisplay];
}
-(void)viewWillDisappear:(BOOL)animated{
    [super viewWillDisappear:animated];
    testTask.taskName = nameTextField.text;
}
-(void)timerAction:(NSTimer *)t
{
    if(testTask.timeInterval == 0)
    {
        if (self.timer)
        {
            [self timerExpired];
            [self.timer invalidate];
            self.timer = nil;
        }
    }
    else
    {
        testTask.timeInterval--;
    }
    NSUInteger seconds = (NSUInteger)round(testTask.timeInterval);
    NSString *string = [NSString stringWithFormat:@"%02u:%02u:%02u",
                        seconds / 3600, (seconds / 60) % 60, seconds % 60];
    timerLabel.text = string;
    NSLog(@"%f", testTask.timeInterval);
}

无论详细视图控制器当前是否在屏幕上,计时器都是支持继续运行。当我第一次点击一个单元格时,转到详细视图控制器并启动计时器,然后返回到tableview,没有问题,倒计时继续进行。问题是,如果我重新点击单元格,则会创建另一个计时器,因此时间减少2倍!不仅如此,显示剩余时间的字符串不会在ViewWillAppear上更新,它只会在第一次调用timerAction时更新。

有很多问题,我真的很感激一些帮助!

2 个答案:

答案 0 :(得分:1)

你在didSelectRowAtIndexPath中提供的代码,我在回答你的另一个问题(one timer per detail view controller)时提供的代码,确实可以防止在按下后退按钮时取消分配控制器。它通过在字典中保持对详细视图控制器的引用来实现此目的。我猜它不适合你,因为你忘了初始化字典,所以它没有。

答案 1 :(得分:0)

即使问题得到解答,我也会发表意见。 一个人,我可能错了,但我不认为防止重新分配是一个好主意,尤其是整个控制器。在iOS中,必须尽快释放内存。

然而,通过查看您的代码,我会做一些事情。

  • 将标签创建移至viewDidLoad,并在viewWillAppear
  • 中更新其内容
  • 在实用程序类中创建和启动计时器,使用适当的方法启动,停止,暂停给定的计时器。该类将确保只有一个(或更多具有指定ID)计时器运行,并且控制器可以请求它。它不是很优雅,但你至少可以使用AppDelegate。