iOS-MRC与ARC区别以及五大内存区

技 术 文 章 / 超 人


2019-03-20 补充@ dynamic与@ synthesize内容


个人觉得要更加深入直观了解MRC与ARC的区别建议先从内存分析开始所以文章开始会从内存说起

文章目录

  • 1.五大内存区域
    • 1.1 栈区
    • 1.2 堆区
    • 1.3 全局区
    • 1.4 常量区
    • 1.5 代码区
    • 1.6 static静态变量
    • 1.7 extern全局变量
    • 1.8 const常量
  • 2.属性标识符
    • 2.1 @property、@synthesize、@dynamic
    • 2.2 nonatomic与atomic
    • 2.3 strong、weak、retain、assgin、copy、unsafe_unretained
    • 2.4 readOnly、readWrite、getter=、setter=
    • 2.5 __unsafe_unretained、__weak、__strong
  • 3.MRC与ARC区别
    • 3.1 MRC手动内存管理
    • 3.2 ARC自动内存管理
    • 3.3 autoreleasepool自动释放池
  • 4.NSString单独说

1.五大内存区域

栈区,堆区,全局区,常量区,代码区

  • 1.1栈区
    创建临时变量时由编译器自动分配,在不需要的时候自动清除的变量的存储区
    里面的变量通常是局部变量函数参数等。在一个进程中,位于用户虚拟地址空间顶部的是用户栈,编译器用它来实现函数的调用。和堆一样,用户栈在程序执行期间可以动态地扩展和收缩。
@interface TestObject()

@end
@implementation TestObject
- (void)testMethodWithName:(NSString *)name
{
   //方法参数name是一个指针,指向传入的参数指针所指向的对象内存地址。name是在栈中
  //通过打印地址可以看出来,传入参数的对象内存地址与方法参数的对象内存地址是一样的。但是指针地址不一样。
  NSLog(@"name指针地址:%p,name指针指向的对象内存地址:%p",&name,name);


  //*person 是指针变量,在栈中, [Person new]是创建的对象,放在堆中。
  //person指针指向了[Person new]所创建的对象。
  //那么[Person new]所创建的对象的引用计数器就被+1了,此时[Person new]对象的retainCount为1
  Person *person = [Person new];
}

  • 1.2堆区
    就是那些由 new alloc 创建的对象所分配的内存块,它们的释放系统不会主动去管,由我们的开发者去告诉系统什么时候释放这块内存(一个对象引用计数为0时系统就会回销毁该内存区域对象)。一般一个 new 就要对应一个 release。在ARC下编译器会自动在合适位置为OC对象添加release操作。会在当前线程Runloop退出或休眠时销毁这些对象,MRC则需程序员手动释放。
    堆可以动态地扩展和收缩。
//alloc是为Person对象分配内存,init是初始化Person对象。本质上跟[Person new]一样。
Person *person = [[Person alloc] init];
  • 1.3全局/静态存储区
    全局变量静态变量被分配到同一块内存中,在以前的 C 语言中,全局变量又分为初始化的和未初始化的(初始化的全局变量和静态变量在一块区域,
    未初始化的全局变量与静态变量在相邻的另一块区域,
    同时未被初始化的对象存储区可以通过 void* 来访问和操纵,
    程序结束后由系统自行释放),在 C++ 里面没有这个区分了,
    他们共同占用同一块内存区。

  • 1.4常量存储区
    这是一块比较特殊的存储区,他们里面存放的是常量,不允许修改。一般值都是放在这个地方的

  • 1.5代码区
    存放函数的二进制代码

NSString *string1;//string1 这个NSString 类型的指针,未初始化存在于<全局区>的<BBS区>

NSString *string2 = @"1234";//string2 这个NSString类型的指针,已初始化存在与<全局区>的<data数据区>,@“1234”存在与堆区,因为@代表了对象。 

static NSString *string3;//string3 这个NSString 类型的指针存在于<全局区>的<BBS区>

static NSString *string4 = @"1234";//string4 这个NSString类型的指针存在与<全局区>的<data数据区>,@“1234”存在与堆区,因为@代表了对象。stiring2和string4的值地址是一样的

static const NSString *string5 = @"654321";//const 修饰后  string5不能修改值。 其他的与string4一样

- (void)test
{
int  a;//a这个int类型的变量 是存在与<栈区>的
a = 10;//10这个值是存在与 <常量区>的

NSStirng *str;//str这个NSString类型的指针 存在于<栈区>
str = @“1234”;//@“1234”这个@对象存在于 <堆区>

static NSString *str1;//str1这个NSString类型的指针 存在于<全局区>的<BBS区>
static NSString *str2 = @"4321';//str2这个NSString类型的指针 存在于<全局区>的<data区>

NSString *str3;//str3这个NSString类型的指针 存在于<栈区>
str3 = [[NSString alloc]initWithString:@"1234"];//[[NSString alloc]initWithString:@"1234"]这个对象 存在于<堆区>

}

1.6 static静态变量

静态变量有两种

  • 全局静态变量
    优点:不管对象方法还是类方法都可以访问和修改全局静态变量,并且外部类无法调用静态变量,定义后只会指向固定的指针地址,供所有对象使用,节省空间。
    缺点:存在的生命周期长,从定义直到程序结束。
    建议:从内存优化和程序编译的角度来说,尽量少用全局静态变量,因为存在的声明周期长,一直占用空间。程序运行时会单独加载一次全局静态变量,过多的全局静态变量会造成程序启动慢,当然就目前的手机处理器性能,几十几百个估计也影响不大吧。
  • 局部静态变量
    优点:定义后只会存在一份值,每次调用都是使用的同一个对象内存地址的值,并没有重新创建,节省空间,只能在该局部代码块中使用。
    缺点:存在的生命周期长,从定义直到程序结束,只能在该局部代码块中使用。
    建议:局部和全局静态变量从本根意义上没有什么区别,只是作用域不同而已。如果值仅一个类中的对象和类方法使用并且值可变,可以定义全局静态变量,如果是多个类使用并可变,建议值定义在model作为成员变量使用。如果是不可变值,建议使用宏定义
static NSString *name;

1.7 extern全局变量
全局变量有两种

  • 对内的全局变量:没有用extern在.h中修饰的变量,仅定义在.m中让该变量只能在该类使用
    优点:不管对象方法还是类方法都可以访问和修改全局静态变量,并且外部类无法调用静态变量,定义后只会存一份值,供所有对象使用,节省空间。跟全局静态变量一样,只是少了static修饰少了static特性
    缺点:存在的生命周期长,从定义直到程序结束
    建议:跟全局静态变量都一样了,还需要用对内的全局变量吗?不用extern修饰就少了extern的特性,还不如用全局静态变量,至少能明确的知道static是对内使用的
  • 外部全局变量:除了该类,其他文件也可以访问该变量
    优点:除了该类,其他文件也可以访问该变量
    缺点:存在的生命周期长,从定义直到程序结束。并且外部可以修改其值,出现错误不容易定位
    建议:使用全局变量的原因就在于其对外的特性,但是其使用的方便性没有使用model的属性或宏来得方便。程序启动的时候会单独加载全局变量,同理与全局静态变量,少使用。
.m中要定义
NSString *name;

.h中同时要定义
extern NSString *name;

全局静态变量与全局变量 其实本质上是没有区别的,只是存在修饰区别,一个static让其只能内部使用,一个extern让其可以外部使用


1.8 const常量
不同于变量,常量的值是固定不可变的,一般用于只读值。
优点:只可以读取值,不能修改。一般用于接口或者文字显示这种固定值。添加extern可以对外全局常量,任意位置都可以访问
缺点:存在的生命周期长,从定义直到程序结束。需要在.h .m中分别定义代码较多
建议:良好的编码习惯而言,少使用宏,多使用常量。因为常量声明是有明确类型的,而宏只是替换并不能进行类型判断。不够严谨。

.h中定义extern
extern NSString *const name;
.m中定义值
NSString *const name = @"123";

//const声明部位不同,意义也不同。const定义的是其右边整体不可变。


//*const name1定义的是 name1 不可变。 name1是指针。
//因此 不能通过修改name1而指向其他值。常规的const使用这个方法定义不可修改的值
NSString *const name1 = @"456";

//const * name定义的是 * name不可变, 而*name指向的是@"123",
//也就是说@"123"这个内存地址的值不可变为其他值。
NSString const * name2 = @"789";
//但name指针可以指向其他值,所以该定义方式无法保证值的唯一性。
NSString *newName = @"222";
name2 = newName;


2.属性标识符

  • 2.1 @property、@synthesize、@dynamic

@synthesize:作用于在@implementation内部,用@synthesiz声明的属性在编译的时候会自动为该属性按照固有规则生成相应的getter setter方法。如果有手动生成getter setter方法也不会报错。

//array是声明的属性名称,_array是编译器自动生成的成员变量。
@synthesize array = _array;

@dynamic:作用于在@implementation内部,与@synthesize不同,使用@dynamic声明时相当于告诉编译器getter setter方法由用户自己生成。如果声明为@dynamic而没有手动生成getter setter方法编译的时候不报错,但是在运行时如果使用.语法去调用该属性时会崩溃。之所以在运行时才会发生崩溃是因为OC具有动态绑定特性。只有在运行时才会去确认具体的调用方法。

@interface TestObject()
{
   NSMutableDictionary *_dic;
}
@property (nonatomic, strong) NSMutableDictionary *dic;
@end
//不同于@ synthesize, @dynamic只需要声明属性名称即可,不需要dic = _dic把值赋值过去。但是需要自己在@interface中声明成员变量。
@dynamic dic;

@property:相对于@dynamic 和 @synthesize ,@property声明的作用区域在@interface内部。 它会告诉编译器自动生成getter setter方法。也允许用户手动生成getter setter中的一个方法,用@property声明的属性不能手动同时写getter setter方法,否则编译器会报错。@property更好的声明属性变量。因为访问方法的命名约定,可以很清晰的看出getter和setter的用处,会传递一些额外信息,后面会跟相应的各种信息例如:@property (nonatomic, strong,onlyread) NSString *name;大多数时候都用的@property声明

@interface TestObject()
{
   NSMutableDictionary *_dic;
}
//因为使用了@property声明,编译器会自动生成相应getter setter方法。
//使用@property不能手动同时生成getter  setter方法,编译器会报错
//nonatomic表示属性是非原子操作,strong表示强引用,
//readonly表示该属性权限为仅读,那么编译器只会生成getter方法,不会生成setter方法
@property (nonatomic, strong,readonly) NSString *name;

@property (nonatomic, strong) NSMutableArray *array;

@property (nonatomic, strong) NSMutableDictionary *dic;

@end
@implementation TestObject

//如果上面的array使用的@property声明,而用户又要手动同时生成getter  setter方法
//可以使用@synthesize 告诉编译器 该属性getter setter方法如果没有手动声明就自动创建,有就不自动生成。
@synthesize array = _array;

//如果dic用@property声明过了,会自动生成getter  setter方法。但是又不希望它自动生成getter setter方法。
//可以用@dynamic 声明。告诉编译器 该属性的getter  setter方法不自动生成
//但如果要自己生成getter setter必须在@ interface内部声明对应的成员变量
@dynamic dic;


- (void)setArray:(NSMutableArray *)array
{
    _array = array;
}

- (NSMutableArray *)array
{
    if (!_array) {
        _array = [NSMutableArray new];
    }
    return _array;
}
- (void)setArray:(NSMutableDictionary *)dic
{
    _dic = dic;
}

- (NSMutableDictionary *)dic
{
    if (!_dic) {
        _dic = [NSMutableDictionary new];
    }
    return _dic;
}

@end
  • 2.2 nonatomic与atomic

nonatomic(非原子性):在调用用nonatomic声明的对象属性时是非线程安全性的。最为直观的就是NSMutableArray的使用。当同时在子线程去增删数组元素,在主线程中去遍历数组元素就会出现数组越界或者数组没有遍历完。因为采用的nonatomic,不同操作可以同时执行,而不需要等前面的操作完成后在进行下一步操作。所以称之为非线程安全。非原子性的执行效率更高不会阻塞线程

atomic(原子性):相反与非原子性,atomic是具有线程安全性的。当属性声明为atomic时,调用该属性的getter/setter方法,会加上同步锁,保证同一时刻只能有一个线程调用属性的getter/setter方法。但是atomic只能保证线程的读和写过程是可靠的。但并不能保证数据一定是可靠的。例如:有线程A和线程B的情况下,从调度顺序来说,线程A先调用了setter方法修改了属性值,然后线程B也调用了setter方法再次修改了属性值,最后线程A调用getter方法获取属性值时,获取到的结果值可能是线程A修改的值,也可能是线程B修改的值,因为A、B是两个线程异步的,最后A调用getter获取的值,取决于B线程是已经设置成功还是B线程还没有执行setter方法。

  • 2.3 strong、weak、retain、assgin、copy、unsafe_unretained

retain:释放旧对象,提高输入对象的引用计数+1,将输入对象的值赋值于旧对象,只能用户声明OC对象

@property (nonatomic, retain) Room *room;
- (void)setRoom:(Room *)room // room = r
{
    // 只有房间不同才需用release和retain
    if (_room != room) {  
        // 将以前的房间释放掉 -1,将旧对象释放
        [_room release];

        // MRC中需要手动对房间的引用计数器+1
        [room retain];

        _room = room;
    }
}

strong:强引用,它是ARC特有。在MRC时代没有,相当于retain。由于MRC时代是靠引用计数器来管理对象什么时候被销毁所以用retain,而ARC时代管理对象的销毁是有系统自动判断,判断的依据就是该对象是否有强引用对象。如果对象没有被任何地方强引用就会被销毁。所以在ARC时代基本都用的strong来声明代替了retain。只能用于声明OC对象(ARC特有)
苹果官网对strong的解释代码:

Precondition:object is a valid pointer to a __strong object which is adequately aligned for a pointer. value is null or a pointer to a valid object.

Performs the complete sequence for assigning to a __strong object of non-block type [[*]]. Equivalent to the following code:

void objc_storeStrong(id *object, id value) {
  id oldValue = *object;
  value = [value retain];
  *object = value;
  [oldValue release];
}

assgin:简单的赋值操作,不会更改引用计数,用于基本的数据类型声明。

weak:弱引用,表示该属性是一种“非拥有关系”。为这种属性设置新值时既不会保留新值也不会释放旧值,类似于assgin。 然而在对象被摧毁时,属性也会被清空(nil out)。这样可以有效的防止崩溃(因为OC中给没有对象地址的指针发送消息不会崩溃,而给有内存地址但地址中是空对象的指针发消息会崩溃,野指针),该声明必须作用于OC对象。
对于 weak 对象会放入一个 hash 表中, 用 weak 指向的对象内存地址作为 key,当此对象的引用计数为0的时候会调用 dealloc, 使用key在weak 表中搜索,将找到的所有对象设置为 nil。(ARC特有),strong 和 weak的指针,根本区别在于,strong执行了retain操作,而weak没有。

runtime 如何实现 weak 变量的自动置 nil?
runtime会对注册的类进行内存布局,对于 weak 修饰的对象会放入一个 hash 表中。
1.初始化时:runtime 会调用 objc_initWeak 函数,初始化一个新的 weak 指针指向对象的地址。
2.添加引用时:objc_initWeak 函数会调用 objc_storeWeak() 函数, objc_storeWeak()的作用是更新指 针指向,创建对应的弱引用表。
3.释放时,调用 clearDeallocating 函数。clearDeallocating 函数首先根据对象地址获取所有 weak 指针 地址的数组,然后遍历这个数组把其中的数据设为 nil,最后把这个 entry 从 weak 表中删除,最后清理对象的记录。

苹果官网对weak的说明
id objc_storeWeak(id *object, id value);
Precondition: object is a valid pointer which either contains a null pointer or has been registered as a __weak >object. value is null or a pointer to a valid object.

If value is a null pointer or the object to which it points has begun deallocation, object is assigned null and >unregistered as a __weak object. Otherwise, object is registered as a __weak object or has its registration >updated to point to value.

Returns the value of object after the call.

copy:不同于其他声明,copy会根据声明的属性是否是可变类型而进行不同操作。如果对象是一个不可变对象,例如NSArray NSString 等,那么copy等同于retain、strong。如果对象是一个可变对象,例如:NSMutableArray,NSMutableString等,它会在内存中重新开辟了一个新的内存空间,用来 存储新的对象,和原来的对象是两个不同的地址,引用计数分别为1. 这就是所谓的深拷贝浅拷贝,浅拷贝只是copy了对象的内存地址,而深拷贝是重新在内存中开辟新空间,新空间对象值与拷贝对象的值一样。但是是完全不同的2个内存地址。 例如copy修饰的类型为 NSString不可变对象时,copy可以保护其封装性,当赋值对象是一个 NSMutableString 类时(NSMutableString是 NSString 的子类,表示一种可修改其值的字符串),此时若是不用copy修饰拷贝字符串,那么赋值该对象之后,赋值对象字符串的值就可能会在其他地方被修改,修改后赋值后对象也会改变,造成值不对。所以,这时就要拷贝一份“不可变” (immutable)的字符串,确保对象中的字符串值不会无意间变动。只要实现属性所用的对象是“可变的” (mutable),就应该在设置新属性值时拷贝一份。

1.当不可变对象进行copy的时候,本质上只是创建了一个新的指针,指向了copy对象的内存空间。所以新指针与原指针所指向的地址相同
2.当可变对象进行copy操作的时候,由于copy方法返回的对象都是不可变对象,所以对可变对象进行copy操作相当于单层深拷贝,创建了一个新指针,并重新再内存中分配了一个新的对象,但是这个对象是不可变对象,而非原对象的可变对象。所以这是一种单层的深拷贝。
3.不可变对象进行mutableCopy的时候,本质上是进行了深拷贝,创建了新指针,并深拷贝重新开辟了一个新的内存空间放置拷贝的对象。所以指针和指针所指向的内存地址与原对象都不一样。并且新的内存中实际对象是一个可变对象。
4.可变对象进行mutableCopy与不可变对象进行mutableCopy本质是一样的。
5.copy一个可变对象并赋值给一个不可变对象,由于copy方法返回的是不可变对象,因此会对原可变对象进行深拷贝,新的内存地址,新的指针。并且copy的结果是一个不可变对象
6.copy一个不可变对象并复制给一个可变对象,同理与上面5的解答,虽然最终赋值给一个可变的指针,但是内存地址中实际是一个不可变对象。
7.mutableCopy一个不可变对象,并赋值给一个可变对象。对原对象进行深拷贝,由于mutableCopy方法返回的是一个可变对。所以实际内存中是一个可变对象

unsafe_unretained:和weak 差不多,唯一的区别便是,对象即使被销毁,指针也不会自动置空, 对象被销毁后指针指向的是一个无用的内存地址(野地址)。如果对象销毁后后还使用此指针,程序会抛出 BAD_ACCESS 的异常。 所以一般不使用unsafe_unretained。目前我还未在实际项目中使用过该声明。(ARC特有)

  • 2.4 readOnly、readWrite、getter=、setter=

readOnly表示属性仅能读不能设置其值。告诉编译器只生成getter方法不生成setter方法。

readWrite默认值,表示属性可读可写。编译器会自动生成getter setter方法

getter=指定属性gettter方法的方法名

@property (nonatomic, strong, getter=getMyDic) NSMutableDictionary *dic;

setter=指定属性setter方法的方法名

@property (nonatomic, strong,setter=myArray:) NSMutableArray *arr;

2.5__unsafe_unretained、__weak、__strong

__unsafe_unretained

NSMutableArray __unsafe_unretained *array = [[NSMutableArray alloc]init];
[array addObject:@"123"];

使用__unsafe_unretained修饰符的变量与使用__weak修饰符的变量一样,因为自己生成并持有的对象不能继续为自己持有,所以生成的对象会立即被释放。也就是说在执行完init方法以后,对象指针所指向的内存就已经释放掉了,但因为用的__unsafe_unretained修饰指针并没不像__weak的指针那样,将指针自动置为nil,它依然指向原来的地址,可是这块地址的内存已经被系统回收了,再访问就是非法的,也就是野指针,再执行后面的addObject方法自然会出错了。

__weak
主要用于解决循环引用,用__weak修饰的变量 当对象释放后,指针自动设置为nil,当后面继续使用该指针变量的时候不会造成crash,更不会造成强引用使该释放的对象无法释放,造成内存泄露。

__weak typeof(self) weakSelf = self;

__strong
相反与__weak,主要用于当使用某个对象时,希望它没有提前被释放。强引用该对象使其无法释放。例如在block内部,希望block调用时该对象不会被提前释放造成错误。可以使用强引用。

TestAlertView *alertView = [TestAlertView new];
alertView = ^()
{
  //当block内部需要使用本身这个局部对象时,需要用强引用方式,让alertView在传递完block后不会被释放依然可以执行setTitle操作
   __strong typeof(alertView) strongAlertView = alertView;
  [strongAlertView setTitle:@"1234"];

}
[alertView show];

3 MRC与ARC区别

3.1 MRC手动内存管理

引用计数器:在MRC时代,系统判定一个对象是否销毁是根据这个对象的引用计数器来判断的。
1.每个对象被创建时引用计数都为1
2.每当对象被其他指针引用时,需要手动使用[obj retain];让该对象引用计数+1。
3.当指针变量不在使用这个对象的时候,需要手动释放release这个对象。 让其的引用计数-1.
4.当一个对象的引用计数为0的时候,系统就会销毁这个对象。

    NSMutableArray *array = [NSMutableArray array];//[NSMutableArray array]创建后引用计数器为1
    NSLog(@"array的对象地址:%p,array的retainCount:%zd",array,[array retainCount]);
    [array release];//调用release后[NSMutableArray array]创建的对象引用计数-1.

     //当程序执行到[array addObject:@"1234"];这里是就会崩溃。因为此时array指针指向的内存地址中没有任何对象,该指针是一个野指针。
     //因为release后[NSMutableArray array]创建的对象引用计数变为了0.系统就会销毁这个内存地址的对象。
    [array addObject:@"1234"];
    NSLog(@"array的对象地址:%p,array的retainCount:%zd",array,[array retainCount]);
    NSLog(@"%@",array);

在MRC模式下必须遵循谁创建,谁释放,谁引用,谁管理
在MRC下使用ARC
在Build Phases的Compile Sources中选择需要使用MRC方式的.m文件,然后双击该文件在弹出的会话框中输入 -fobjc-arc

3.2 ARC自动内存管理
WWDC2011和iOS5所引入自动管理机制——自动引用计数(ARC),它不是垃圾回收机制而是编译器的一种特性。ARC管理机制与MRC手动机制差不多,只是不再需要手动调用retain、release、autorelease;当你使用ARC时,编译器会在在适当位置插入release和autorelease;ARC时代引入了strong强引用来带代替retain,引入了weak弱引用。
在ARC下使用MRC方法
在ARC工程中如果要使用MRC的需要在工程的Build Phases的Compile Sources中选择需要使用MRC方式的.m文件,然后双击该文件在弹出的会话框中输入 -fno-objc-arc


在非MRC文件中无法使用retain release retainCount 方法,无法再dealloc方法中调用[super dealloc];方法

3.3 autoreleasepool自动释放池

自动释放池始于MRC时代,主要是用于 自动 对 释放池内 对象 进行引用计数-1的操作,即自动执行release方法。

autorelease、autoreleasepool、release
autorelease:本质是将对象放入当前的自动释放池中
autoreleasepool:本质是当前runloop销毁自动释放池时会自动对自动释放池内的对象调用release
release:是对当前对象的引用计数进行减1的操作,如果对象引用计算为0,则会释放对象的内存空间。

autoreleasepool的原理
autoreleasepool本质上是一个__AtAutoreleasePool的结构体,通过__AtAutoreleasePool的构造函数和析构函数去实现对池内的对象进行自动释放。

1.当对象调用autorelease方法时,会将对象加入autoreleasepool的栈中,而autoreleasepool会在__AtAutoreleasePool结构体的构造函数中调用autoreleasepoolPage::push方法将自动释放池中的对象加入到双向链表中,每个autoreleasepoolpage大小都是4096个字节。所以如果autoreleasepool中对象超过4096个字节。那么就会再创建一个autoreleasepoolpage。因此autoreleasepoolpage有多个。
2.当__AtAutoreleasePool的结构体被销毁时,会自动调用析构函数,通过析构函数调用autoreleasepoolPage::pop方法会向栈中的对象发送release消息

在MRC中使用autoreleasepool必须在代码块内部手动为对象调用autorelease把对象加入到的自动释放池,系统会自动在代码块结束后,对加入自动释放池中的对象发送一个release消息。无需手动调用release

int main(int argc, const char * argv[])
{
Person *father =  [[Person alloc] init];//引用计数为1
    @autoreleasepool {//这里创建自动释放池
        int a = 10; // a在栈,10在常量区
        int b = 20; //b在栈,20在常量区
        // p : 栈
        // [[Person alloc] init]创建的对象(引用计数器==1) : 在堆
        Person *p = [[Person alloc] init];
       //MRC下需要手动让对象加入自动释放池
       [p autorelease];

        Person *pFather = father;
        [father retain];//father指针指向的对象被pFather引用,在MRC下需要手动让被引用对象引用计数+1
        
        NSLog(@"pFather对象内存地址:%p,pFather的引用计数:%zd",pFather,[pFather retainCount]);
        NSLog(@"father对象内存地址:%p,father的引用计数:%zd",father,[father retainCount]);

    }//这里释放 自动释放池
    // 当autoreleasepool内部代码块执行完毕后上面代码块后, 栈里面的变量a、b、p 都会被回收
    // 但是堆里面的Person对象还会留在内存中,因为它是计数器依然是1。当autoreleasepool代码块执行完毕后,会对释放池内部的所有对象执行一个release消息。如果发送release消息后,对象引用计数为0了,那么就会被系统回收。


NSLog(@"father对象内存地址:%p,father的引用计数:%zd",father,[father retainCount]);
    return 0;
}

在ARC中对@autoreleasepool的使用相比MRC不太多。主要用于一些大内存消耗对象的重复创建时,保证内存处于比较优越的状态。常用于创建对象较多的for循环中。在ARC下不要手动的为@autoreleasepool代码块内部对象添加autorelease,ARC下自动的把@autoreleasepool代码块中创建的对象加入了自动释放池中。

    for (int i = 0; i < 10000000; i++)
    {
        @autoreleasepool{
            NSMutableArray *array = [NSMutableArray new];
            NSMutableDictionary *dic = [NSMutableDictionary new];
            NSMutableArray *array1 = [NSMutableArray new];
            NSMutableDictionary *dic1 = [NSMutableDictionary new];
            NSMutableArray *array2 = [NSMutableArray new];
            NSMutableDictionary *dic2 = [NSMutableDictionary new];
            NSData *data = UIImageJPEGRepresentation([UIImage imageNamed:@"testimage"], 1);
            NSError *error;
            NSURL *url = [NSURL URLWithString:@"www.baidu.com"];
            NSString *fileContents = [NSString stringWithContentsOfURL:url
                                                              encoding:NSUTF8StringEncoding
                                                                 error:&error];
        }
    }
使用autoreleasepool时

可以从上图看出,使用autoreleasepool时,内存一直保持稳定状态,上次几乎没浮动

不使用autoreleasepool

因为没有使用 autoreleasepool,随着for循环一直执行下去,内存一直在上升。

autorelease对象释放的时机
1.系统自行创建的@autoreleasepool释放时机
线程与Runloop是一对一关系,主线程中会自动创建Runloop,而子线程需要自行调用Runloop来让子线程自动创建Runloop。当@autoreleasepool加入到某个线程时,该线程的Runloop会
借用runloop的Autorelease对象释放的背后的解释(ARC环境下)

Runloop

图中第1步 Observer 监视的事件是 Entry(即将进入Loop),其回调内会调用 _objc_autoreleasePoolPush() 创建自动释放池。其 order 是-2147483647,优先级最高,保证创建释放池发生在其他所有回调之前

图中第6步 Observer 监视了两个事件: BeforeWaiting(准备进入休眠) 时调用_objc_autoreleasePoolPop() 和 _objc_autoreleasePoolPush() 释放旧的池并创建新池;Exit(即将退出Loop) 时调用 _objc_autoreleasePoolPop() 来释放自动释放池。这个 Observer 的 order 是 2147483647,优先级最低,保证其释放池子发生在其他所有回调之后。

图中第10 Observer 监视事件是exit(即讲退出runloop),其回调内会调用 _objc_autoreleasePoolpop() 释放自动释放池。

从上面就能看出,Runloop中系统自动创建的@autoreleasepool是在准备进入休眠状态才被销毁的。所以在ARC下,在线程中的临时对象是在当前线程的Runloop进入休眠或者退出loop或者退出线程时被执行release的。

2.自己创建的@autoreleasepool

@autoreleasepool
{//这个{开始创建的自动释放池,这里开始内部的对象自动加入autorelease


}//这个}开始,不必再对 对象加入autorelease,自动释放池被销毁。

从上面就能看出,自行创建的@autoreleasepool,是在}后被释放的,而其中的autorelease对象,也是在这个时候自动执行的release操作。

1, 2 两点可以看出,虽然autorelease对象释放的时机并不都是在代码块结束后就释放。但是他们有一个共同特性,那就是必定是在@autoreleasepool被销毁时,释放的。 所以要清楚autorelease对象什么时候被释放,只需要搞清楚@autoreleasepool什么时候被销毁即可

主线程中既有系统创建的@autoreleasepool也有开发者自行创建的@autoreleasepool。那么他的释放顺序是怎样的呢?
因为@autoreleasepool是以栈的形式存储的,按照先进后出的规则释放栈中每个@autoreleasepool。主线程的@autoreleasepool是在Runloop一开始就创建了所以,它必然是栈最里面的,而自行创建的@autoreleasepool是在Runloop运行中创建的,所以在栈上面一点。按照栈的规则,@autoreleasepool是先释放自行创建的@autoreleasepool,在释放系统创建的。


NSString 单独说

为什么要单独说NSString呢,因为NSString在内存上与其他类型存在很大的不同

    //该代码是在MRC环境下测试用  

    NSString *str1 = @"123456789";//用@""方法创建一个  固定长度为9的字符串
    NSString *str2 = @"1234567890";//用@""方法创建一个  固定长度为10的字符串
    
    
    NSString *str3 = [NSString stringWithFormat:@"234567890"];//用stringWithFormat方法创建一个  固定长度为9的字符串
    NSString *str4 = [NSString stringWithFormat:@"2345678901"];//用stringWithFormat方法创建一个  固定长度为10的字符串
    
    NSString *str5 = [[NSString alloc] initWithString:@"345678901"];//用initWithString方法创建一个  固定长度为9的字符串
    NSString *str6 = [[NSString alloc] initWithString:@"3456789012"];//用initWithString方法创建一个  固定长度为9的字符串
    
    NSString *str7 = [[NSString alloc] initWithFormat:@"456789012"];//用initWithFormat方法创建一个  固定长度为9的字符串
    NSString *str8 = [[NSString alloc] initWithFormat:@"4567890123"];//用initWithFormat方法创建一个  固定长度为9的字符串
    
    NSString *str9 = [NSString stringWithFormat:@"1234567890"];//用stringWithFormat方法创建一个  固定长度为10的字符串并与str2字符串一样的字符串
    NSString *str10 = [[NSString alloc] initWithString:@"1234567890"];//用initWithString方法创建一个  固定长度为10的字符串并与str2字符串一样的字符串
    
    NSLog(@"str1 用@"" 的retainCount为:%ld \n  对象内地地址:%p",[str1 retainCount],str1);
    NSLog(@"str2 用@"" 的retainCount为:%ld \n  对象内地地址:%p",[str2 retainCount],str2);
    NSLog(@"-----------------------------------------------------------------");
    NSLog(@"str3 用stringWithFormat 的retainCount为:%ld \n  对象内地地址:%p",[str3 retainCount],str3);
    NSLog(@"str4 用stringWithFormat 的retainCount为:%ld \n  对象内地地址:%p",[str4 retainCount],str4);
    NSLog(@"-----------------------------------------------------------------");
    NSLog(@"str5 用initWithString 的retainCount为:%ld \n  对象内地地址:%p",[str5 retainCount],str5);
    NSLog(@"str6 用initWithString 的retainCount为:%ld \n  对象内地地址:%p",[str6 retainCount],str6);
    NSLog(@"-----------------------------------------------------------------");
    NSLog(@"str7 用initWithFormat 的retainCount为:%ld \n  对象内地地址:%p",[str7 retainCount],str7);
    NSLog(@"str8 用initWithFormat 的retainCount为:%ld \n  对象内地地址:%p",[str8 retainCount],str8);
    NSLog(@"-----------------------------------------------------------------");
    NSLog(@"使用lu看一下 str1 的retainCount为:%lu \n  对象内地地址:%p",[str1 retainCount],str1);
    NSLog(@"使用lu看一下 str4 的retainCount为:%lu \n  对象内地地址:%p",[str4 retainCount],str4);
    NSLog(@"-----------------------------------------------------------------");
    NSLog(@"str9  字符串与str2一样 的retainCount为:%lu \n  对象内地地址:%p",[str9 retainCount],str9);
    NSLog(@"str10  字符串与str2一样 的retainCount为:%lu \n  对象内地地址:%p",[str10 retainCount],str10);


输出结果


结果

1.为什么用 @"", stringWithFormat, initWithString, initWithFormat四种方式。

因为方式不同,创建的对象所存在内存区域不同。你会发现str2 与 str9 字符串内容一样。为什么对象地址一个是0x1053674f8 另一个是0x604000033580。正常情况下,字符串内容一样,应该取的是同一个内存地址。就像str2与str10一样。虽然创建的方法不一样,但是字符串一样,内存地址就是一样的。 这就是区别。 @“”与initWithString方法创建的字符串于stringWithFormat、initWithFormat创建的字符串所在的内存区域不同。

2.为什么要区分长度9 和 长度10的字符串?

因为字符串长度不一样,字符串所在内存区域不同,retainCount也不同。从上面结果中可以看出,str3和str4是同一种方式创建的字符串,但一个内存是0xa287dcaecc2ac5d9一个是0x6040000339a0。因为前者是在五大区域之外的内存区,而后者在堆中。

由上面连2点结合可知,由initWithString和stringWithString创建的NSString对象,不管字符串的内容和长度怎么变化,该字符串对象始终是存储在常量区的,引用计数为-1;从用%lu打印来看initWithString和stringWithString创建的字符串retainCount是无符号长整型的最大值。所以可以说他们没有引用计数这个概念

而由initWithFormat和stringWithFormat创建的对象,如果字符串内容是非汉字的,那么当字符串长度小于10个时,该字符串存储区域在五大区域之外,且随着字符串长度的变化,存储地址会有很大变化。当字符串长度超过10个以后,该字符串在堆中,与正常的OC对象一样。这里为什么要说非汉字呢,因为如果字符串内容是汉字,不管字符串的内容和长度怎么变化,该字符串都是在堆中,与正常OC对象一样。

App占用手机内存最大限制记录

型号 最大限制
iPad1 127MB/256MB/49%
iPad2 275MB/512MB/53%
iPad3 645MB/1024MB/62%
iPad4 585MB/1024MB/57% (iOS 8.1)
iPad Mini 1st Generation 297MB/512MB/58%
iPad Mini retina 696MB/1024MB/68% (iOS 7.1)
iPad Air 697MB/1024MB/68%
iPad Air 2 1383MB/2048MB/68% (iOS 10.2.1)
iPad Pro 9.7" 1395MB/1971MB/71% (iOS 10.0.2 (14A456))
iPad Pro 10.5” 3057/4000/76% (iOS 11 beta4)
iPad Pro 12.9” (2015) 3058/3999/76% (iOS 11.2.1)
iPad Pro 12.9” (2017) 3057/3974/77% (iOS 11 beta4)
iPad Pro 11.0” (2018) 2858/3769/76% (iOS 12.1)
iPad Pro 12.9” (2018) 4598/5650/81% (iOS 12.1)
iPod touch 4th gen 130MB/256MB/51% (iOS 6.1.1)
iPod touch 5th gen 286MB/512MB/56% (iOS 7.0)
iPhone4 325MB/512MB/63%
iPhone4s 286MB/512MB/56%
iPhone5 645MB/1024MB/62%
iPhone5s 646MB/1024MB/63%
iPhone6 645MB/1024MB/62% (iOS 8.x)
iPhone6+ 645MB/1024MB/62% (iOS 8.x)
iPhone6s 1396MB/2048MB/68% (iOS 9.2)
iPhone6s+ 1392MB/2048MB/68% (iOS 10.2.1)
iPhoneSE 1395MB/2048MB/69% (iOS 9.3)
iPhone7 1395/2048MB/68% (iOS 10.2)
iPhone7+ 2040MB/3072MB/66% (iOS 10.2.1)
iPhone X 1392/2785/50% (iOS 11.2.1)
iPhone XS Max 2039/3735/55% (iOS 12.1)
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,686评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,668评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,160评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,736评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,847评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,043评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,129评论 3 410
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,872评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,318评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,645评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,777评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,470评论 4 333
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,126评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,861评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,095评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,589评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,687评论 2 351