So *_* It 4 singleton objective-c uitableview ios segue
我很感激任何帮助解决我的问题让我在最后一天拔头发!
免责声明1 我是一个Objective-C菜鸟 - 所以答案可能非常简单(我来自多年的PHP).请友好一点.
免责声明2 是的,我已经广泛"谷歌搜索"并查看了Apple文档 - 但未能理解我的问题.
摘要
我试图将表视图中选择的值从单例传递给另一个控制器.我发现第二个视图控制器中的viewDidLoad方法在tableView:didSelectRowAtIndexPath:方法之前完成,该方法触发了segue.这导致第二个视图控制器访问"陈旧"数据.
详情
我使用针对iOS5的ARC设置了一个非常简单的StoryBoard项目.

我创建了一个单例来在视图控制器之间存储NSString.
Singleton.h
#import <Foundation/Foundation.h>
@interface Singleton : NSObject
@property (strong, nonatomic) NSString *sharedString;
+ (id)singletonManager;
@end
Run Code Online (Sandbox Code Playgroud)
Singleton.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
Run Code Online (Sandbox Code Playgroud)
故事板中的TableView链接到以下TableViewController
TableViewController.h
#import <UIKit/UIKit.h>
@interface TableViewController : UITableViewController
@end
Run Code Online (Sandbox Code Playgroud)
TableViewController.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
Run Code Online (Sandbox Code Playgroud)
表格单元格被推送(推送)到具有单个标签的基本ViewController.
DetailViewController.h
#import <UIKit/UIKit.h>
@interface DetailViewController : UIViewController
@property (strong, nonatomic) IBOutlet UILabel *displayLabel;
@end
Run Code Online (Sandbox Code Playgroud)
DetailViewController.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
Run Code Online (Sandbox Code Playgroud)
当选择表格单元格时,我想要
不幸的是,这与我的代码无法正常工作.如果我在控制台中查看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
Run Code Online (Sandbox Code Playgroud)
Arrrrggghh!
如果我将标签文本的设置放在详细视图的viewDidAppear方法中,则将显示共享单例NSString中的正确值.但是,通过这种方法,您实际上可以看到应用程序更改文本标签的值(例如,如果您单击第一行,当详细视图屏幕加载时,您可以看到标签从"初始化字符串"更改为"第一行" ").
有人可以解释一下我如何将一个共享值从单例传递到另一个视图,并在视图实际呈现到屏幕之前将其提供给第二个视图控制器.
如果需要任何澄清 - 请告诉我.
我确信这与我未能理解视图生命周期的微妙之处有关.
UPDATE
感谢*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);
}
}
Run Code Online (Sandbox Code Playgroud)
在目标视图控制器中设置属性(公共),以便设置要传入的值.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];
}
}
Run Code Online (Sandbox Code Playgroud)
如果您愿意,也可以在此处设置标签文本的值.
viewDidLoad之前完成的原因是iOS在转换到目标视图控制器之前设置了目标视图控制器,使用该prepareForSegue方法可以保证目标视图控制器已经实例化并设置,因此您可以设置所需的值.
| 归档时间: |
|
| 查看次数: |
2503 次 |
| 最近记录: |