連絡先リストから人を選択し、名、姓、メールアドレスを取得するアプリがあります。次に、名をnsmutablearrayに保存し、それをuitableviewセルに配置します。シミュレータで連絡先を選択すると、問題が発生します。
コード:
.h:
#import <UIKit/UIKit.h>
#import <AddressBookUI/AddressBookUI.h>
@interface FirstViewController : UIViewController < ABPeoplePickerNavigationControllerDelegate, UITableViewDelegate, UITableViewDataSource>
- (IBAction)showPicker:(id)sender;
@property (weak, nonatomic) IBOutlet NSString *firstName;
@property (weak, nonatomic) IBOutlet NSString *email;
@property (weak, nonatomic) IBOutlet NSString *lastName;
@property (weak, nonatomic) IBOutlet UITableView *myTableView;
@property (strong, nonatomic) NSMutableArray *contacts;
@end
.m:
#import "FirstViewController.h"
@interface FirstViewController ()
@end
@implementation FirstViewController
@synthesize firstName;
@synthesize email;
@synthesize lastName;
@synthesize contacts;
@synthesize myTableView;
- (void)viewDidLoad
{
[super viewDidLoad];
// Do any additional setup after loading the view, typically from a nib.
contacts = [[NSMutableArray alloc]initWithObjects:nil];
}
- (void)didReceiveMemoryWarning
{
[super didReceiveMemoryWarning];
// Dispose of any resources that can be recreated.
}
#pragma mark - UITableView Datasource
-(NSInteger)numberOfSectionsInTableView:(UITableView *)tableView
{
return 1;
}
-(NSInteger)tableView:(UITableView *)tableView numberOfRowsInSection:(NSInteger)section
{
return contacts.count;
}
-(UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath
{
static NSString *cellIdentifier = @"Cell";
UITableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:cellIdentifier];
if (cell == nil) {
cell = [[UITableViewCell alloc] initWithStyle:UITableViewCellStyleDefault reuseIdentifier:cellIdentifier];
}
cell.textLabel.text = [contacts objectAtIndex:indexPath.row];
return cell;
}
- (IBAction)showPicker:(id)sender {
ABPeoplePickerNavigationController *picker =
[[ABPeoplePickerNavigationController alloc] init];
picker.peoplePickerDelegate = self;
[self presentModalViewController:picker animated:YES];
}
- (void)peoplePickerNavigationControllerDidCancel:
(ABPeoplePickerNavigationController *)peoplePicker
{
[self dismissModalViewControllerAnimated:YES];
}
- (BOOL)peoplePickerNavigationController:
(ABPeoplePickerNavigationController *)peoplePicker
shouldContinueAfterSelectingPerson:(ABRecordRef)person {
[self displayPerson:person];
[self dismissModalViewControllerAnimated:YES];
return NO;
}
- (BOOL)peoplePickerNavigationController:
(ABPeoplePickerNavigationController *)peoplePicker
shouldContinueAfterSelectingPerson:(ABRecordRef)person
property:(ABPropertyID)property
identifier:(ABMultiValueIdentifier)identifier
{
return NO;
}
- (void)displayPerson:(ABRecordRef)person
{
NSString* name = (__bridge_transfer NSString*)ABRecordCopyValue(person,
kABPersonFirstNameProperty);
self.firstName = name;
NSString* last = (__bridge_transfer NSString*)ABRecordCopyValue(person,
kABPersonLastNameProperty);
self.lastName = last;
ABMultiValueRef emails = ABRecordCopyValue(person, kABPersonEmailProperty);
NSString *emailId = (__bridge NSString *)ABMultiValueCopyValueAtIndex(emails, 0);//0 for "Home Email" and 1 for "Work Email".
self.email = emailId;
if (!(contacts))
{
contacts = [[NSMutableArray alloc]init];
}
[contacts insertObject:firstName atIndex:0];
NSIndexPath * indexPath = [NSIndexPath indexPathForRow:0 inSection:0];
[self.myTableView insertRowsAtIndexPaths:@[indexPath] withRowAnimation:UITableViewRowAnimationAutomatic];
}
@end
UITableViewは、常にデータソースと同期している必要があります。データソースがバックグラウンドスレッドで変更される可能性がある場合は、特別な注意が必要です。
データソースに何かが追加されたら、できるだけ早くbeginUpdate/insert/endUpdateを呼び出します。これらのキャッシュについて心配する必要はありません。UITableViewは、十分なCPU時間とリソースがあると判断したときに実行される変更をキャッシュします。
EndUpdatesが呼び出されると、UITableはdataSourceにセクションと行の数を再度要求します。セクションと行の数がdataSourceから直接フィードされる場合、セクションと行の数、挿入、削除の数は、numberOfSectionsとnumberOfRowsInSectionの終了呼び出しによって返される数と等しくなければなりません。
最後のヒント:「reloadData」とbeginUpdate/endUpdateのペアへの呼び出しを混在させないでください。両方ではなく、どちらか一方を使用してください。
これと同じ問題が発生しました。あなたがしなければならないのは変更だけです
[self.myTableView insertRowsAtIndexPaths:@[indexPath]withRowAnimation:UITableViewRowAnimationAutomatic];`
に
[self.myTableView beginUpdates];
[self.myTableView insertRowsAtIndexPaths:@[indexPath]withRowAnimation:UITableViewRowAnimationAutomatic];
[self.myTableView endUpdates];
UITableView
ドキュメントから
beginUpdates
Begin a series of method calls that insert, delete, or select rows and sections of the receiver.
beginUpdates
を使用する場合は、endUpdates
ではなくreloadData
を呼び出す必要があります。
UITableViewの詳細については、 https://developer.Apple.com/library/ios/documentation/uikit/reference/UITableView_Class/Reference/Reference.html を確認できます。
お役に立てれば!
この問題は、ViewController内にテーブルビューを配置するときによく発生します。 UITableViewControllerを使用している場合は、3にジャンプします。
これらの手順が役立つ場合があります。
1:View Controllerの.hファイルに、次のものを追加してください。
@interface YourViewController : UIViewController <UITableViewDataSource, UITableViewDelegate>
2:次に、Ctrlキーを押しながら.hクラスにドラッグして、テーブルビューのIBOutletを作成します。次のようになります。
@property (weak, nonatomic) IBOutlet UITableView *tableView;
3:次のステップは、Ctrlキーを押しながらView Controllersアイコンにドラッグすることです(画像を参照)
これを2回選択して行う必要があります。
- delegate
- datasource
最後に、.mファイルには、次のメソッドが必要です。
- (void) viewWillAppear:(BOOL)animated{
//[self.tableView beginUpdates];
//[self.tableView endUpdates];
[self.tableView reloadData];
}
BeginUpDates/endUpdatesまたはreloadDataのいずれかを使用できますが、AppleドキュメントではreloadDataを推奨しています。
完了すると、テーブルは正常に機能するはずです。
-tableView:numberOfRowsInSection:
の実装に関する上記のコメントは正しいです。アサーションは、戻り値が増加することを期待して戻り値をチェックしています。
ただし、dataSource
にUITableView
を設定しなかったため、存在しないメソッドを呼び出して(または呼び出しずに)0
を取得します。
myTableView.dataSource
と(デリゲートプロトコルも実装しているため)myTableView.delegate
をself
に設定する必要があります。
また、次のようなものが必要になる可能性があります
- (void)viewDidLoad
{
[super viewDidLoad];
[self.myTableView registerClass:[UITableViewCell class] forCellReuseIdentifier:@"Cell"];
...
}
他の場所で登録する場合、またはストーリーボードに、コードが要求する識別子"Cell"
の「プロトタイプセル」がある場合を除きます。
連絡先配列のカウントを維持するおよびそれに応じてインクリメントする
そしてindexPathの作成中に、次のことを行う必要があります適切なindexPathForRowを設定します:およびセクション数(必要な場合)
- (void)displayPerson:(ABRecordRef)person
{
..........
[contacts insertObject:firstName atIndex:0];
NSIndexPath * indexPath = [NSIndexPath indexPathForRow:0 inSection:0]; // you cant do this increment appropriately
.........
}
テーブルビューのデータソースとデリゲートメソッドを確認してください。空の配列をデータソースメソッドに渡している可能性があります。また、データを取得した後、tableviewをリロードすることを忘れないでください。
私はそれが愚かであることを知っています-デリゲートとdataSourceを設定するのを忘れたので、同じエラーが発生しました。
したがって、行を挿入してtableView.endUpdates()
を実行した後、tableViewはいくつかの行が必要であると考えましたが、リンクされていないdataSource
のため、そうではありませんでした。