[英]Problems passing data to a DetailViewController from TableView in iOS5
我很感激任何幫助解決我的問題讓我在最后一天拔頭發!
免責聲明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
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
故事板中的TableView鏈接到以下TableViewController
TableViewController.h
#import <UIKit/UIKit.h>
@interface TableViewController : UITableViewController
@end
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
表格單元格被推送(推送)到具有單個標簽的基本ViewController。
DetailViewController.h
#import <UIKit/UIKit.h>
@interface DetailViewController : UIViewController
@property (strong, nonatomic) IBOutlet UILabel *displayLabel;
@end
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
當選擇表格單元格時,我想要
不幸的是,這與我的代碼無法正常工作。 如果我在控制台中查看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
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);
}
}
在目標視圖控制器中設置屬性(公共),以便設置要傳入的值。在源視圖控制器中實現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
方法可以保證目標視圖控制器已經實例化並設置,因此您可以設置所需的值。
聲明:本站的技術帖子網頁,遵循CC BY-SA 4.0協議,如果您需要轉載,請注明本站網址或者原文地址。任何問題請咨詢:yoyou2525@163.com.