为啥我的可变数组和表视图行为异常?
Posted
技术标签:
【中文标题】为啥我的可变数组和表视图行为异常?【英文标题】:Why are my mutable arrays and table views behaving unexpectedly?为什么我的可变数组和表视图行为异常? 【发布时间】:2013-06-12 15:18:35 【问题描述】:我正在构建一个登录应用。它有两个表视图。一个列出之前访问过的人(existingNames),一个列出当前登录的人(姓名)。
在我的代码中的某些点,唯一不会在访问时使程序崩溃的可变数组是名称。
names 和 existingNames 似乎也以某种方式被颠倒了。当我尝试从名称中删除时,程序崩溃。当我从现有名称中删除时,更改会反映在 tableView2 中,但 tableView2 应该与名称相关联。
在应用程序的当前状态下,除了访问任何公司数组外,一切都在“工作”。由于名称和现有名称被向后使用,所以我在工作中加上引号。
任何对可能导致问题的原因的见解将不胜感激!
.h:
#import <UIKit/UIKit.h>
@interface ViewController : UIViewController <UITableViewDataSource, UITableViewDelegate>
UITableView *tableView;
@property (weak, nonatomic) IBOutlet UITextField *nameField;
@property (weak, nonatomic) IBOutlet UITextField *companyField;
@property (nonatomic, retain) NSMutableArray *names;
@property (nonatomic, retain) NSMutableArray *companies;
@property (nonatomic, retain) NSMutableArray *existingNames;
@property (nonatomic, retain) NSMutableArray *existingCompanies;
- (IBAction)add:(id)sender;
- (IBAction)addExisting:(id)sender;
- (IBAction)remove:(id)sender;
- (IBAction)submit:(id)sender;
@property (strong, nonatomic) IBOutlet UITableView *tableView1;
@property (weak, nonatomic) IBOutlet UITableView *tableView2;
@end
.m:
#import "ViewController.h"
#import <MessageUI/MessageUI.h>
@interface ViewController () <MFMailComposeViewControllerDelegate>
@end
@implementation ViewController
@synthesize nameField;
@synthesize companyField;
@synthesize names;
@synthesize companies;
@synthesize existingNames;
@synthesize existingCompanies;
@synthesize tableView1 = _tableView1;
@synthesize tableView2 = _tableView2;
int rowNumber1;
int rowNumber2;
- (void)viewDidLoad
[super viewDidLoad];
self.names = [[NSMutableArray alloc] init];
self.companies = [[NSMutableArray alloc] init];
self.existingNames = [[NSMutableArray alloc] init];
self.existingCompanies = [[NSMutableArray alloc] init];
// Do any additional setup after loading the view, typically from a nib.
- (void)didReceiveMemoryWarning
[super didReceiveMemoryWarning];
// Dispose of any resources that can be recreated.
- (NSInteger)tableView:(UITableView *)tableView numberOfRowsInSection:(NSInteger)section
if (tableView == self.tableView1)
return [existingNames count];
else if (tableView == self.tableView2)
return [names count];
return 0;
- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath
static NSString *simpleTableIdentifier = @"SimpleTableItem";
UITableViewCell *cell;
if (tableView == self.tableView1)
cell = [_tableView1 dequeueReusableCellWithIdentifier:simpleTableIdentifier];
else if (tableView == self.tableView2)
cell = [_tableView2 dequeueReusableCellWithIdentifier:simpleTableIdentifier];
if (cell == nil)
cell = [[UITableViewCell alloc] initWithStyle:UITableViewCellStyleDefault reuseIdentifier:simpleTableIdentifier];
if (tableView == self.tableView1)
cell.textLabel.text = [existingNames objectAtIndex:indexPath.row];
else if (tableView == self.tableView2)
cell.textLabel.text = [names objectAtIndex:indexPath.row];
return cell;
- (void)tableView:(UITableView *)tableView didSelectRowAtIndexPath:(NSIndexPath *)indexPath
if (tableView == self.tableView1)
rowNumber1 = indexPath.row;
else if (tableView == self.tableView2)
rowNumber2 = indexPath.row;
- (IBAction)add:(id)sender
BOOL exists = [existingNames containsObject:nameField.text];
if(exists == FALSE)
[names addObject:nameField.text];
[companies addObject:companyField.text];
[existingNames addObject:nameField.text];
[existingCompanies addObject:companyField.text];
else
[names addObject:nameField.text];
[companies addObject:companyField.text];
[_tableView1 reloadData];
[_tableView2 reloadData];
nameField.text=@"";
companyField.text=@"";
- (IBAction)addExisting:(id)sender
[existingNames addObject:[names objectAtIndex:rowNumber1]];
//[companies addObject:[existingCompanies objectAtIndex:rowNumber]];
[_tableView2 reloadData];
- (IBAction)remove:(id)sender
[existingNames removeObjectAtIndex:rowNumber2];
[existingCompanies removeObjectAtIndex:rowNumber2];
[_tableView2 reloadData];
@end
【问题讨论】:
【参考方案1】:以下方法是UITableViewDataSource
和UITableViewDelegate
协议的一部分
tableView:numberOfRowsInSection:
tableView:cellForRowAtIndexPath:
tableView:didSelectRowAtIndexPath:
并且将由您的表格视图调用,假设您已为表格视图的delegate
和dataSource
属性分配了属性。然而,这些方法,
tableView2:numberOfRowsInSection:
tableView2:cellForRowAtIndexPath:
tableView2:didSelectRowAtIndexPath:
不是协议的一部分,不会被表视图调用。看起来您可能会混淆您的属性名称,例如tableView
,带有协议方法名称,例如tableView:numberOfRowsInSection:
。你需要做的是:
-
如果您还没有这样做,请将您的视图控制器设置为
delegate
和 dataSource
用于 tableView
和 tableView2
。
在您需要实现的每个数据源和委托方法中,处理两个表的情况。
例如,您的 tableView:numberOfRowsInSection:
方法如下所示:
-(NSInteger)tableView:(UITableView *)tableView numberOfRowsInSection:(NSInteger)section
if (tableView == self.tableView)
return [existingNames count];
else if (tableView == self.tableView2)
return [names count];
return 0;
【讨论】:
我已经为两个表视图设置了数据源和委托。感谢您在理解 tableview 文档方面的帮助,他们真的有很多东西可以吸收!我会努力的,如果有任何改进,我会通知你。 我已经有一段时间了,我已经让程序运行了,只是现在表格视图没有被填充。我正在发布我的新代码。我假设它与 tableview 方法有关。每次我在这些方法中引用 tableView 时都会收到警告“tableView 的本地声明隐藏实例变量”。我已经尝试了其他方法来消除此警告,但没有其他方法起作用。 啊,我通过在方法声明中添加 _ 来摆脱它(即 - (NSInteger)tableView:(UITableView *)_tableView numberOfRowsInSection:(NSInteger)section)。我的表格视图目前仍未被填充。有什么建议吗? 我假设它现在与我在这些方法中调用 tableView 和 _tableView 时有关。我无法理解何时使用哪个。 在您的界面中,您已经定义了一个变量“tableView”和一个属性“tableView”。但是在您的实现中,您正在综合和引用“tableView1”,您似乎与您的变量不一致。我建议a)删除“tableView”变量定义,b)将属性从“tableView”重命名为“tableView1”,c)撤消“将_添加到方法声明”,d)看看事情的样子。您可以在委托方法中放置日志语句或断点,以确保它们被调用、验证您的数组、已填充等。以上是关于为啥我的可变数组和表视图行为异常?的主要内容,如果未能解决你的问题,请参考以下文章
在可变大小的字符串数组中,如果输入来自 0 索引 A[0] 不可访问;如果输入来自索引 1,A[0] 和 A[1] 不可行为啥?
为啥 C# 数组对 Enumeration 使用引用类型,而 List<T> 使用可变结构?