ios基础之 透过页面跳转来认识 Strong 与 Weak
最近在自己做一个小程序,遇到了页面跳转的问题,然后上网一通乱搜,跳转的问题解决了,又有传值的问题。上面两个问题解决了,又发现内存比刚开始时多占用了2M,于是,各种内心纠结,想彻底
搞清楚strong 和 weak 在ARC下到底是怎么个意思,也顺便理清了页面跳转之间的一些联系。
下面开始进入正题:(程序使用了storyboard)
由于页面中使用了一个动态加载的tableview,所以没法在页面中拖segue来实现页面跳转,那么只能在代码中实现页面跳转了。
下面上一下页面跳转的小片段:
页面A的控制器代码 头文件
1 @interface BKViewController : UIViewController<UITableViewDelegate,UITableViewDataSource> 2 @property (weak, nonatomic) IBOutlet UITableView *tableview1; 3 4 @property (strong, nonatomic) UIViewController *checkFT; 5 @property (weak, nonatomic) UIViewController *checkFTWeak; 6 @end
头文件中定义了3个属性,
第一个属性就是对应页面上的一个tableview没什么好说的了。
第二个属性:一个strong 的页面B的控制器变量
第三个属性:一个weak 的页面B的控制器变量
具体干什么,稍后再说
1 @interface BKViewController () 2 { 3 BOOL _isRegistNib; 4 NSDateFormatter *_shortDateFormatter; 5 NSArray *tableData; 6 7 NSArray *kpType;//记账类型 8 int pushCount;//纪录跳转次数 9 } 10 @end 11 12 @implementation BKViewController 13 14 15 - (void)viewDidLoad { 16 [super viewDidLoad]; 17 // Do any additional setup after loading the view. 18 UIBarButtonItem *btnRight = [[UIBarButtonItem alloc] initWithTitle:@"测试" style:UIBarButtonItemStylePlain target:self action:@selector(test)]; 19 [self.navigationItem setRightBarButtonItem:btnRight]; 20 21 _isRegistNib = NO; 22 _shortDateFormatter = [DateFormatterHelper getShortDateFormatter]; 23 //资金类型数据加载 24 kpType = [NSArray arrayWithObjects:@"记账日期",@"记账类型",@"资金流动类型",@"费用科目",nil]; 25 [self loadTableViewData]; 26 self.tableview1.dataSource = self; 27 self.tableview1.delegate = self; 28 29 pushCount = 1; 30 }
上面这段代码主要关注pushCount这个变量,它记录从A页面跳往B页面的次数,初始化为1
ok,准备工作完成了,下面说一下要实验的步骤:
1)从页面A 跳转到 页面B,跳转之前,将weak的 checkFTWeak 指针指向 页面B的对象;
2)然后回到页面A
3)点击测试按钮,测试一下 checkFTWeak现在指向的对象是否还存在
4)我们要有一个笼统的概念:strong会持有对象(引用计数+1),weak不会
代码:
1 -(void)tableView:(UITableView *)tableView didSelectRowAtIndexPath:(NSIndexPath *)indexPath 2 { 3 /* 4 LycTableCellViewDefault *cell = (LycTableCellViewDefault *)[tableView cellForRowAtIndexPath:indexPath];*/ 5 6 NSLog(@"当前跳转次数%i",pushCount); 7 //跳转到选择资金类型页面 8 if (indexPath.section == 0 && indexPath.row == 1) { 9 UIStoryboard *sbMain = [UIStoryboard storyboardWithName:@"Main" bundle:nil]; 10 CheckFlowTypeViewController *cv = [sbMain instantiateViewControllerWithIdentifier:@"checkFlowTypeSB"]; 11 12 if (pushCount == 1) { 13 self.checkFTWeak = cv; 14 } 15 NSLog(@"checkFTWeak指向的对象是%@", self.checkFTWeak); 16 pushCount++; 17 //[self presentViewController:cv animated:YES completion:nil ]; 18 [self.navigationController pushViewController: cv animated:YES]; 19 } 20 } 21 22 -(void)test 23 { 24 NSLog(@"checkFTWeak指向的对象是%@", self.checkFTWeak); 25 }
首先初始化要跳转到的页面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,打印两个指针指向的对象
修改后的代码:
1 -(void)tableView:(UITableView *)tableView didSelectRowAtIndexPath:(NSIndexPath *)indexPath 2 { 3 /* 4 LycTableCellViewDefault *cell = (LycTableCellViewDefault *)[tableView cellForRowAtIndexPath:indexPath];*/ 5 6 NSLog(@"当前跳转次数%i",pushCount); 7 //跳转到选择资金类型页面 8 if (indexPath.section == 0 && indexPath.row == 1) { 9 UIStoryboard *sbMain = [UIStoryboard storyboardWithName:@"Main" bundle:nil]; 10 CheckFlowTypeViewController *cv = [sbMain instantiateViewControllerWithIdentifier:@"checkFlowTypeSB"]; 11 self.checkFT = cv; 12 13 if (pushCount == 1) { 14 self.checkFTWeak = cv; 15 } 16 NSLog(@"checkFT指向的对象是%@",self.checkFT); 17 NSLog(@"checkFTWeak指向的对象是%@", self.checkFTWeak); 18 pushCount++; 19 //[self presentViewController:cv animated:YES completion:nil ]; 20 [self.navigationController pushViewController: cv animated:YES]; 21 } 22 } 23 24 -(void)test 25 { 26 NSLog(@"点击测试按钮后:"); 27 NSLog(@"checkFT指向的对象是%@",self.checkFT); 28 NSLog(@"checkFTWeak指向的对象是%@", self.checkFTWeak); 29 }
运行的结果:
第一次跳转因为有strong类型的指针checkFT给weak类型的指针checkFTWeak撑腰,所以他俩指向对象一样。
在返回页面A后,发生第二次跳转钱checkFT指向了新的页面B的对象,checkFTWeak指向的对象被释放了,所以打印出了nil
那么又得出了第三个结论:
当strong类型的指针从指向a变成指向b之后,原来的a对象的引用计数将会-1,如果没有其他指针持有它,它变被arc给释放了。然后b的引用计数会+1;
分析结束,如果不对之处希望指出共同学习。