4

最後の日、髪の毛を抜いてしまった問題を解決するのに助けていただければ幸いです。

免責事項1 私はObjective-Cの初心者なので、これに対する答えは非常に単純かもしれません(私は何年ものPHPから来ています)。よろしくお願いします。

免責事項2 はい、私は広範囲にわたって「グーグル」してAppleのドキュメントを調べましたが、私の問題を理解することができませんでした。

概要

テーブルビューで選択した値をシングルトンから別のコントローラーに渡そうとしています。セグエをトリガーしたtableView:didSelectRowAtIndexPath:メソッドの前に、2番目のビューコントローラーのviewDidLoadメソッドが完了していることがわかりました。これにより、2番目のViewControllerアクセス​​の「古い」データが発生します。

詳細

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

テーブルセルを選択したとき、欲しい

  • 共有シングルトンNSStringに値を設定するTableViewController
  • 詳細ビューへのセグエを開始します
  • DetailViewControllerを使用して、共有シングルトンNSStringの値を取得し、ラベルに表示します
  • 画面にレンダリングする詳細ビュー

残念ながら、これは私のコードでは期待どおりに機能していません。コンソールでNSLogの詳細を見ると、テーブルビューメソッド( didSelectRowAtIndexPath )を開始する前に、詳細ビューのviewDidLoad関数が実際に終了していることがわかります。

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からの正しい値が表示されます。ただし、このアプローチでは、アプリがテキストラベルの値を変更しているのを実際に確認できます(たとえば、最初の行をクリックすると、詳細ビ​​ュー画面が読み込まれたときに、ラベルが「初期化文字列」から「最初」に変更されているのを確認できます。 ")。

誰かが、共有値をシングルトンから別のビューに渡し、ビューが実際に画面にレンダリングされる前に、それを2番目のビューコントローラーで使用できるようにする方法を説明できますか。

説明が必要な場合は、お知らせください。

これは、ビューのライフサイクルの微妙さを理解できなかったことと関係があると確信しています。

アップデート

*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

デスティネーションビューコントローラで、渡す値を設定できるようにプロパティ(public)を設定し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 に答える