最近在自己做一个小程序,遇到了页面跳转的问题,然后上网一通乱搜,跳转的问题解决了,又有传值的问题。上面两个问题解决了,又发现内存比刚开始时多占用了2M,于是,各种内心纠结,想彻底
搞清楚strong 和 weak 在ARC下到底是怎么个意思,也顺便理清了页面跳转之间的一些联系。
下面开始进入正题:(程序使用了storyboard)
由于页面中使用了一个动态加载的tableview,所以没法在页面中拖segue来实现页面跳转,那么只能在代码中实现页面跳转了。
下面上一下页面跳转的小片段:
页面A的控制器代码 头文件
@interface BKViewController : UIViewController<UITableViewDelegate,UITableViewDataSource>
@property (weak, nonatomic) IBOutlet UITableView *tableview1; @property (strong, nonatomic) UIViewController *checkFT;
@property (weak, nonatomic) UIViewController *checkFTWeak;
@end
头文件中定义了3个属性,
第一个属性就是对应页面上的一个tableview没什么好说的了。
第二个属性:一个strong 的页面B的控制器变量
第三个属性:一个weak 的页面B的控制器变量
具体干什么,稍后再说
@interface BKViewController ()
{
BOOL _isRegistNib;
NSDateFormatter *_shortDateFormatter;
NSArray *tableData; NSArray *kpType;//记账类型
int pushCount;//纪录跳转次数
}
@end @implementation BKViewController - (void)viewDidLoad {
[super viewDidLoad];
// Do any additional setup after loading the view.
UIBarButtonItem *btnRight = [[UIBarButtonItem alloc] initWithTitle:@"测试" style:UIBarButtonItemStylePlain target:self action:@selector(test)];
[self.navigationItem setRightBarButtonItem:btnRight]; _isRegistNib = NO;
_shortDateFormatter = [DateFormatterHelper getShortDateFormatter];
//资金类型数据加载
kpType = [NSArray arrayWithObjects:@"记账日期",@"记账类型",@"资金流动类型",@"费用科目",nil];
[self loadTableViewData];
self.tableview1.dataSource = self;
self.tableview1.delegate = self; pushCount = ;
}
上面这段代码主要关注pushCount这个变量,它记录从A页面跳往B页面的次数,初始化为1
ok,准备工作完成了,下面说一下要实验的步骤:
1)从页面A 跳转到 页面B,跳转之前,将weak的 checkFTWeak 指针指向 页面B的对象;
2)然后回到页面A
3)点击测试按钮,测试一下 checkFTWeak现在指向的对象是否还存在
4)我们要有一个笼统的概念:strong会持有对象(引用计数+1),weak不会
代码:
-(void)tableView:(UITableView *)tableView didSelectRowAtIndexPath:(NSIndexPath *)indexPath
{
/*
LycTableCellViewDefault *cell = (LycTableCellViewDefault *)[tableView cellForRowAtIndexPath:indexPath];*/ NSLog(@"当前跳转次数%i",pushCount);
//跳转到选择资金类型页面
if (indexPath.section == && indexPath.row == ) {
UIStoryboard *sbMain = [UIStoryboard storyboardWithName:@"Main" bundle:nil];
CheckFlowTypeViewController *cv = [sbMain instantiateViewControllerWithIdentifier:@"checkFlowTypeSB"]; if (pushCount == ) {
self.checkFTWeak = cv;
}
NSLog(@"checkFTWeak指向的对象是%@", self.checkFTWeak);
pushCount++;
//[self presentViewController:cv animated:YES completion:nil ];
[self.navigationController pushViewController: cv animated:YES];
}
} -(void)test
{
NSLog(@"checkFTWeak指向的对象是%@", self.checkFTWeak);
}
首先初始化要跳转到的页面B对象 cv,然后将 checkFTWeak 指针指向 cv;接着就打印出对象的信息
当从B 跳回 到A时,点击A页面的测试按钮,再打印 checkFTWeak指针,看看它指向的对象还存在否
通过运行结果,我们看到,当页面从B 返回 A 之后,checkFTWeak指向的对象,也就是那个页面b不存在了。
那么如果让checkFT这个strong类型的属性指向页面b是什么结果呢?
修改tableview选中事件的那句代码为:self.checkFT = cv;
运行结果为:
可以看到,由于checkFT指针是strong类型的属性,那么他会持有页面B的对象,导致内存不会释放
结合上面两个例子,可以总结下面两点:
1)页面跳转时创建的目标页面(就是页面B),在返回后(也就是调用popViewController方法),将会被释放
2)明白了strong和weak的区别,
strong指向的对象不会被释放,除非把指针设置为nil;
而weak不会对指向的对象的引用计数有任何影响,在饮用对象不存在时,会返回nil,替我们做了处理;
那么到这里我还有一个疑问:如果strong类型的指针,本来指向 c,后来又指向d的话,那么c会被释放还是继续存在?
那么继续来改造代码,这个时候前面提到的pushCount参数就起作用了,具体试验步骤如下:
1)从页面A跳转到页面B,同时让checkFT、 checkFTWeak 两种类型的指针指向页面B的对象
2)返回后打印两个指针指向的对象
3)再次从页面A跳转到页面B,但是strong类型的 checkFT指针指向新初始化的页面B,checkFTWeak指针还是指向原来的那个对象
4)返回页面A,打印两个指针指向的对象
修改后的代码:
-(void)tableView:(UITableView *)tableView didSelectRowAtIndexPath:(NSIndexPath *)indexPath
{
/*
LycTableCellViewDefault *cell = (LycTableCellViewDefault *)[tableView cellForRowAtIndexPath:indexPath];*/ NSLog(@"当前跳转次数%i",pushCount);
//跳转到选择资金类型页面
if (indexPath.section == && indexPath.row == ) {
UIStoryboard *sbMain = [UIStoryboard storyboardWithName:@"Main" bundle:nil];
CheckFlowTypeViewController *cv = [sbMain instantiateViewControllerWithIdentifier:@"checkFlowTypeSB"];
self.checkFT = cv; if (pushCount == ) {
self.checkFTWeak = cv;
}
NSLog(@"checkFT指向的对象是%@",self.checkFT);
NSLog(@"checkFTWeak指向的对象是%@", self.checkFTWeak);
pushCount++;
//[self presentViewController:cv animated:YES completion:nil ];
[self.navigationController pushViewController: cv animated:YES];
}
} -(void)test
{
NSLog(@"点击测试按钮后:");
NSLog(@"checkFT指向的对象是%@",self.checkFT);
NSLog(@"checkFTWeak指向的对象是%@", self.checkFTWeak);
}
运行的结果:
第一次跳转因为有strong类型的指针checkFT给weak类型的指针checkFTWeak撑腰,所以他俩指向对象一样。
在返回页面A后,发生第二次跳转钱checkFT指向了新的页面B的对象,checkFTWeak指向的对象被释放了,所以打印出了nil
那么又得出了第三个结论:
当strong类型的指针从指向a变成指向b之后,原来的a对象的引用计数将会-1,如果没有其他指针持有它,它变被arc给释放了。然后b的引用计数会+1;
分析结束,如果不对之处希望指出共同学习。