ios基础之 透过页面跳转来认识 Strong 与 Weak

时间:2022-08-08 23:11:44

最近在自己做一个小程序,遇到了页面跳转的问题,然后上网一通乱搜,跳转的问题解决了,又有传值的问题。上面两个问题解决了,又发现内存比刚开始时多占用了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指针,看看它指向的对象还存在否

ios基础之 透过页面跳转来认识 Strong 与 Weak

ios基础之 透过页面跳转来认识 Strong 与 Weak

ios基础之 透过页面跳转来认识 Strong 与 Weak

通过运行结果,我们看到,当页面从B 返回 A 之后,checkFTWeak指向的对象,也就是那个页面b不存在了。

那么如果让checkFT这个strong类型的属性指向页面b是什么结果呢?

修改tableview选中事件的那句代码为:self.checkFT = cv;

运行结果为:

ios基础之 透过页面跳转来认识 Strong 与 Weak

可以看到,由于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);
}

运行的结果:

ios基础之 透过页面跳转来认识 Strong 与 Weak

第一次跳转因为有strong类型的指针checkFT给weak类型的指针checkFTWeak撑腰,所以他俩指向对象一样。

在返回页面A后,发生第二次跳转钱checkFT指向了新的页面B的对象,checkFTWeak指向的对象被释放了,所以打印出了nil

那么又得出了第三个结论:

当strong类型的指针从指向a变成指向b之后,原来的a对象的引用计数将会-1,如果没有其他指针持有它,它变被arc给释放了。然后b的引用计数会+1;

分析结束,如果不对之处希望指出共同学习。