首先,我想说我只是提出这个问题,因为我想了解发生了什么。
我在Xcode5上的全新安装中打开了一个旧的Xcode项目(一个非常简单的项目)。 当我意识到它在iOS 7上不起作用时。为什么?不知道..
我看到了其他一些问题,没有得到任何有用的答案,所以我做了一个简单的测试。
在UITableViewController
所有工作正常,但didSelectRowAtIndexPath
检查出来
RootViewController.h:
@interface RootViewController : UITableViewController
@property (strong, nonatomic) NSMutableArray *items;
@end
RootViewController.m 在viewDidLoad中我初始化数组(带有一些字符串)。
实现dataSource和委托方法(是的,我将委托和dataSource设置为tableView)
- (NSInteger)numberOfSectionsInTableView:(UITableView *)tableView
{
return 1;
}
- (NSInteger)tableView:(UITableView *)tableView numberOfRowsInSection:(NSInteger)section
{
return [_items 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];
}
// Configure the cell...
cell.textLabel.text = [_items objectAtIndex:indexPath.row];
return cell;
}
问题在于:
- (void)tableView:(UITableView *)tableView didSelectRowAtIndexPath:(NSIndexPath *)indexPath
{
// Navigation logic may go here, for example:
// Create the next view controller.
DetailViewController *detailViewController = [[DetailViewController alloc] initWithNibName:@"DetailViewController" bundle:nil];
// Push the view controller.
[self.navigationController pushViewController:detailViewController animated:NO];
detailViewController.detailLabel.text = [_items objectAtIndex:indexPath.row];
}
DetailViewController
是一个简单的UIViewController:
(是的,我在nib文件上设置了IBOutlet)
@interface DetailViewController : UIViewController
@property (weak, nonatomic) IBOutlet UILabel *detailLabel;
@end
问题是这在iOS7上不起作用,DetailViewController
中的标签不会更新。我尝试在pushViewController
之前和之后设置标签文本。
适用于所有以前版本的iOS。
为什么不在iOS 7上工作?
我开始工作的唯一方法就像我在this other question上看到的那样:
- (void)tableView:(UITableView *)tableView didSelectRowAtIndexPath:(NSIndexPath *)indexPath
{
// Navigation logic may go here, for example:
// Create the next view controller.
DetailViewController *detailViewController = [[DetailViewController alloc] initWithNibName:@"DetailViewController" bundle:nil];
// Push the view controller.
[self.navigationController pushViewController:detailViewController animated:YES];
dispatch_async(dispatch_get_main_queue(), ^{
detailViewController.detailLabel.text = [_items objectAtIndex:indexPath.row];
});
}
有人可以帮助我了解这里发生了什么吗?
谢谢!
_
修改
它也是这样的:
- (void)tableView:(UITableView *)tableView didSelectRowAtIndexPath:(NSIndexPath *)indexPath
{
// Navigation logic may go here, for example:
// Create the next view controller.
DetailViewController *detailViewController = [[DetailViewController alloc] initWithNibName:@"DetailViewController" bundle:nil];
// Push the view controller.
[self.navigationController pushViewController:detailViewController animated:YES];
if (detailViewController.view) {
// do notihing
}
detailViewController.detailLabel.text = [_items objectAtIndex:indexPath.row];
}
答案 0 :(得分:8)
当您设置UILabel的值时,视图尚未加载,因此如果您检入调试,您可能会发现detailLabel为nil。
为什么不在自定义init方法中传递值? e.g。
- (id)initWithDetails:(NSString *)detailsText;
然后在viewDidLoad中,分配值?
根据您的编辑,回答您关于其工作原理的问题:
- (void)tableView:(UITableView *)tableView didSelectRowAtIndexPath:(NSIndexPath *)indexPath
{
TestViewController *test = [[testViewController alloc] init];
NSLog(@"Before: %@", test.label);
if(test.view){}
NSLog(@"After: %@", test.label);
[[test label] setText:@"My Test"];
[self.navigationController pushViewController:test animated:YES];
}
我已经模拟了你的场景,如果你在ViewController上访问'view',如果视图不存在,它将调用viewDidLoad。所以你的UILabel现在已经定了。这是控制台的输出。
2013-10-11 16:21:04.555 test[87625:11303] Before: (null)
2013-10-11 16:21:04.559 test[87625:11303] After: <UILabel: 0x75736b0; frame = (148 157; 42 21); text = 'Label'; clipsToBounds = YES; opaque = NO; autoresize = RM+BM; userInteractionEnabled = NO; layer = <CALayer: 0x7573740>>
我建议使用自定义init方法,而不是这种方法 - 上面的例子是一个黑客。使用类似这样的东西。
- (void)tableView:(UITableView *)tableView didSelectRowAtIndexPath:(NSIndexPath *)indexPath
{
testViewController *test = [[testViewController alloc] initWithText:@"My Test"];
[self.navigationController pushViewController:test animated:YES];
}
如果您不想使用自定义初始化程序,我建议您在DetailSController上使用NSString类型的公共属性。在调用init之后使用您的方法分配。然后在DetailViewController中,在viewDidLoad中,或者出现,将IBOutlet设置为包含NSString的属性的值。
答案 1 :(得分:0)
由于你正在写作,它没有得到更新 @property(弱,非原子)IBOutlet UILabel * detailLabel; 你正在将UILabel作为弱财产。你必须把它变成强大的财产。 @property(强,非原子)IBOutlet UILabel * detailLabel; 当你使你的财产变弱以致它被摧毁时。
要了解更多关于弱势和强势资产的信息,您可以参考 Differences between strong and weak in Objective-C