iOS 内存管理主要依赖于 自动引用计数(ARC,Automatic Reference Counting) 机制。ARC 通过自动管理对象的引用计数来确保对象在不再需要时被释放,从而避免内存泄漏和野指针问题。
1. 内存管理的基本概念
1.1 引用计数(Reference Counting)
引用计数是一种内存管理技术,每个对象都有一个计数器,记录有多少个指针指向它。当引用计数为 0 时,对象会被释放。
- retain:增加引用计数。
- release:减少引用计数。
- autorelease:将对象加入自动释放池,稍后减少引用计数。
1.2 自动引用计数(ARC)
ARC 是编译器特性,编译器会在编译时自动插入 retain
、release
和 autorelease
代码,开发者无需手动管理引用计数。
2. ARC 的工作原理
2.1 强引用(Strong Reference)
默认情况下,对象的引用是强引用,会增加引用计数。只要有一个强引用指向对象,对象就不会被释放。
NSString *str = [[NSString alloc] initWithFormat:@"Hello"];
// str 是一个强引用,引用计数为 1
2.2 弱引用(Weak Reference)
弱引用不会增加引用计数,常用于避免循环引用。当对象被释放时,弱引用会自动置为 nil
。
__weak NSString *weakStr = str;
// weakStr 是一个弱引用,引用计数仍为 1
2.3 自动释放池(Autorelease Pool)
自动释放池用于延迟释放对象,通常在事件循环结束时释放池中的对象。
@autoreleasepool {
NSString *tempStr = [NSString stringWithFormat:@"Temporary"];
// tempStr 会被加入自动释放池
}
// 自动释放池结束时,tempStr 会被释放
3. 循环引用与解决
循环引用指两个或多个对象相互持有强引用,导致引用计数无法降为 0,从而引发内存泄漏。
3.1 使用弱引用打破循环
通过将其中一个引用改为弱引用,可以打破循环引用。
__weak typeof(self) weakSelf = self;
self.block = ^{
[weakSelf doSomething];
};
3.2 使用 __block
修饰符
在 MRC 中,__block
修饰符可以打破循环引用,但在 ARC 中,__block
也会增加引用计数,因此仍需结合弱引用使用。
4. 内存管理实践
4.1 避免强引用循环
在涉及 Block、Delegate、KVO 等场景时,需特别注意避免循环引用。
4.2 使用 Instruments 检测内存泄漏
Xcode 的 Instruments 工具可以帮助检测内存泄漏和循环引用。
4.3 合理使用自动释放池
在循环中创建大量临时对象时,使用自动释放池及时释放内存。
for (int i = 0; i < 10000; i++) {
@autoreleasepool {
NSString *tempStr = [NSString stringWithFormat:@"Temp %d", i];
// 使用 tempStr
}
}
5. MRC 与 ARC 的区别
- MRC(Manual Reference Counting):开发者需手动管理引用计数。
- ARC:编译器自动管理引用计数,开发者只需关注对象的所有权关系。
6. 总结
iOS 内存管理主要依赖 ARC,通过引用计数自动管理对象生命周期。开发者需理解强引用、弱引用和自动释放池的使用,避免循环引用和内存泄漏。合理使用工具检测内存问题,确保应用性能。