4

我将不胜感激任何帮助解决我在最后一天把头发拉出来的问题!

免责声明 1 我是一个 Objective-C 菜鸟 - 所以这个问题的答案可能很简单(我来自 PHP 多年)。请友好一点。

免责声明 2 是的,我已经广泛地“谷歌搜索”并查看了 Apple 文档 - 但未能理解我的问题。

概括

我正在尝试将在表视图中选择的值从单例传递给另一个控制器。我发现第二个视图控制器中的viewDidLoad方法在触发 segue的tableView: didSelectRowAtIndexPath:方法之前完成。这导致第二个视图控制器访问“陈旧”数据。

细节

我已经使用面向 iOS5 的 ARC 建立了一个非常简单的 StoryBoard 项目。

故事板

我创建了一个单例来存储视图控制器之间的 NSString。

单例.h

#import <Foundation/Foundation.h>
@interface Singleton : NSObject
@property (strong, nonatomic) NSString *sharedString;
+ (id)singletonManager;
@end

单身人士.m

#import "Singleton.h"

static Singleton *sharedInstance = nil;

    @implementation Singleton
    @synthesize sharedString = _sharedString;
    +(id)singletonManager
    {
        @synchronized(self){
            if (sharedInstance == nil)
                sharedInstance = [[self alloc] init];
        }
        return sharedInstance;
    }
    -(id) init {
        if (self = [super init]) {
            _sharedString = [[NSString alloc] initWithString:@"Initialization String"];
        }
        return self;
    }

    @end

故事板中的 TableView 链接到下面的 TableViewController

表视图控制器.h

#import <UIKit/UIKit.h>

@interface TableViewController : UITableViewController
@end

表视图控制器.m

#import "TableViewController.h"
#import "Singleton.h"

@interface TableViewController ()
@property NSArray *tableData;
@end

@implementation TableViewController
@synthesize tableData;

- (id)initWithStyle:(UITableViewStyle)style
{
    self = [super initWithStyle:style];
    if (self) {
    }
    return self;
}

- (void)viewDidLoad
{

    [super viewDidLoad];
    self.tableData = [NSArray arrayWithObjects:@"First", @"Second", @"Third", nil];
}

- (void)viewDidUnload
{
    [super viewDidUnload];
}


- (NSInteger)numberOfSectionsInTableView:(UITableView *)tableView
{
    // Return the number of sections.
    return 1;
}

- (NSInteger)tableView:(UITableView *)tableView numberOfRowsInSection:(NSInteger)section
{
    // Return the number of rows in the section.
    return [self.tableData count];
}

- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath
{
    static NSString *CellIdentifier = @"Cell";
    UITableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:CellIdentifier];

    // Configure the cell...
    cell.textLabel.text = [self.tableData objectAtIndex:indexPath.row];
    return cell;
}

- (void)tableView:(UITableView *)tableView didSelectRowAtIndexPath:(NSIndexPath *)indexPath
{    
    NSLog(@"TableViewController: didSelectRowAtIndexPath start");
    Singleton *sharedInstance = [Singleton singletonManager];
    sharedInstance.sharedString = [self.tableData objectAtIndex:indexPath.row];
    NSLog(@"TableViewController: Finished storing data into shared string: %@", sharedInstance.sharedString);
    NSLog(@"TableViewController: didSelectRowAtIndexPath start");
}

@end

表格单元格被 segue-ed(推送)到具有单个标签的基本 ViewController。

DetailViewController.h

#import <UIKit/UIKit.h>

@interface DetailViewController : UIViewController
@property (strong, nonatomic) IBOutlet UILabel *displayLabel;

@end

细节视图控制器.m

#import "DetailViewController.h"
#import "Singleton.h"

@interface DetailViewController ()

@end

@implementation DetailViewController
@synthesize displayLabel = _displayLabel;

- (void)viewDidLoad
{
    [super viewDidLoad];
    NSLog(@"DetailViewController: viewDidLoad start");
    Singleton *sharedInstance = [Singleton singletonManager];
    self.displayLabel.text = sharedInstance.sharedString;
    NSLog(@"displayLabel.text: %@", self.displayLabel.text);
    NSLog(@"DetailViewController: viewDidLoad end");
}

- (void)viewDidUnload
{
    [self setDisplayLabel:nil];
    [super viewDidUnload];
}

@end

选择表格单元格时,我想要

  • TableViewController 在共享单例 NSString 中设置一个值
  • 开始转到详细视图
  • DetailViewController 获取共享单例 NSString 中的值并将其显示在标签中
  • 要渲染到屏幕的详细视图

不幸的是,这与我的代码没有按预期工作。如果我查看控制台中的 NSLog 详细信息,我可以看到详细视图的viewDidLoad函数实际上是在启动表视图方法(didSelectRowAtIndexPath)之前完成的。

2012-05-10 23:17:45.756 TestSingleton[12105:f803] DetailViewController: viewDidLoad start
2012-05-10 23:17:45.758 TestSingleton[12105:f803] displayLabel.text: Initialization String
2012-05-10 23:17:45.759 TestSingleton[12105:f803] DetailViewController: viewDidLoad end
2012-05-10 23:17:45.763 TestSingleton[12105:f803] TableViewController: didSelectRowAtIndexPath start
2012-05-10 23:17:45.764 TestSingleton[12105:f803] TableViewController: Finished storing data into shared string: First
2012-05-10 23:17:45.765 TestSingleton[12105:f803] TableViewController: didSelectRowAtIndexPath start

啊啊啊啊!

如果我将标签文本的设置放在详细视图的viewDidAppear方法中,将显示来自共享单例 NSString 的正确值。但是,使用这种方法,您实际上可以看到应用程序更改了文本标签的值(例如,如果您单击第一行,当详细视图屏幕加载时,您可以看到标签从“初始化字符串”更改为“第一行” ”)。

有人可以解释一下我如何将共享值从单例传递到另一个视图,并在视图实际呈现到屏幕之前将其提供给第二个视图控制器。

如果需要任何澄清 - 请告诉我。

我确信这与我未能欣赏视图生命周期的微妙之处有关。

更新

感谢 *8vius。

我在TableViewController.m中的新方法解决了我的问题:

- (void) prepareForSegue:(UIStoryboardSegue *)segue sender:(id)sender
{
    if([segue.identifier isEqualToString:@"mySegue"]) {
        NSLog(@"Preparing for Segue to detail view");
        Singleton *sharedInstance = [Singleton singletonManager];
        NSIndexPath *selectedRow = [self.tableView indexPathForSelectedRow];
        sharedInstance.sharedString = [self.tableData objectAtIndex:selectedRow.row];
        NSLog(@"TableViewController: Finished storing data into shared string: %@", sharedInstance.sharedString);

    }
}
4

1 回答 1

6

在您的目标视图控制器中设置一个属性(公共),以便您可以设置您希望传递的值。prepareForSegue在您的原始视图控制器中实现并导入目标视图控制器的标题。

要获取表格视图单元格的 indexPath,您应该将 IBOutlet 设置为表格视图,这样您可以使用 prepareForSegue 方法中的 indexPathForCell 方法获取 indexPath。

在你的prepareForSegue你应该有这样的东西:

- (void)prepareForSegue:(UIStoryboardSegue *)segue sender:(id)sender {
  NSIndexPath *indexPath = [self.tableView indexPathForCell:sender];
  if ([segue.identifier isEqualToString:@"[identifier for your segue]"]) {
     [segue.destinationViewController setValueIWantToSet:value];
  }
}

如果您愿意,还可以在此处设置标签文本的值。

viewDidLoad之前完成的原因是因为iOS在转换到它之前设置了目标视图控制器,使用该prepareForSegue方法可以保证你的目标视图控制器已经实例化和设置,所以你可以设置你想要的值。

于 2012-05-10T14:04:59.403 回答