美文网首页
关于ARC内存管理

关于ARC内存管理

作者: 赖晓嘉LMT | 来源:发表于2018-06-08 10:46 被阅读0次

ARC是编译器(时)特性,而不是运行时特性,更不是垃圾回收器(GC)。

OC是使用引用计数机制(retainCount)来管理内存。内存每被引用一次,该内存的引用计数+1,每被释放一次引 用计数-1。当引用计数= 0 的时候,调用该对象的 dealloc 方法,来彻底从内存中删除该对象。

alloc,allocWithZone,new(带初始化)时:该对象引用计数 +1;

retain:手动为该对象引用计数 +1;(使用set方法赋值时,实质上是会先保留新值,再释放旧值,再设置新值,避免新旧值一样时导致对象被释放的的问题)

copy:对象引用计数 +1;(使用set方法赋值时,实质上是会先拷贝新值,再释放旧值,再设置新值)

mutableCopy:生成一个新对象,新对象引用计数为 1;

release:手动为该对象引用计数 -1;

autorelease:把该对象放入自动释放池,当自动释放池释放时,其内的对象引用计数 -1;(ARC下可以通过__autoreleasing修饰符进行标记,否则的话看方法名, 非alloc/new/copy/mutableCopy开头的方法编译器都会自动帮我们调用autorelease方法.)

ARC 下涉及到的 一些 修饰符

1.关键字 __strong  默认值

* 在程序中定义的引用,默认就是强引用,所谓的强引用指向一个对象时,对象的引用计数器会自动加1,当引用超出作用域,对象的引用计数器自动减1,

* 定义强引用:__Strong Student* stu = [[Student alloc] init];

* 使用set方法赋值时,实质上是会先保留新值,再释放旧值,再设置新值,避免新旧值一样时导致对象被释放的的问题。

* 当一个对象被引用指向时,此对象会隐式的retain一次,当强引用超出作用域时,指向的对象会隐式的release一次

* 引用在使用的时候,会根据作用域的范围,自动做加1减1操作

2.关键字__weak 弱引用

__weak Student* stu = [[Student alloc] init];

* 使用set方法赋值时,实质上不保留新值,也不释放旧值,只设置新值。

* 仅仅就是指向对象,

* 当一个弱引用指向的对象被销毁时,弱引用本身会自动的赋值为nil

3.关键字 __autoreleasing 用于标示自动释放的对象(变量)

* 在对象(变量)被标记之后,ARC会将变量放到自动释放池中(@autoreleasepool)等自动释放池销毁时(自动释放池的销毁时间是确定的,一般是在程序事件处理之后释放,或者由我们自己手动释放),会给标记的对象发送release消息,将其引用计数减1(如果自动释放池向对象发送release消息后对象的引用计数仍大于1,对象就无法销毁),并回收内存

4.__unsafe_unretained 不安全的弱引用,若没有任何强引用指针指向该变量,不会自动设为空,会成为野指针(为了兼容iOS5以下版本的产物,可以理解成MRC下的weak,现在基本用不到)。

注:@autoreleasepool实质上是一个__AtAutoreleasePool的结构体对象;这个结构体会在初始化时调用objc_autoreleasePoolPush()方法,会在析构时调用objc_autoreleasePoolPop()方法(关键点:入栈,出栈,双向链表)。这个地方有详细解释

那么在内存管理中也是有一些规则的:

1)当你通过new、alloc或copy方法创建一个对象时,它的引用计数为1,当不再使用该对象时,应该向对象发送release或者autorelease消息释放对象()。

2)当你通过其他方法获得一个对象时,如果对象引用计数为1且被设置为autorelease,则不需要执行任何释放对象的操作;

3)如果你打算取得对象所有权,就需要保留对象并在操作完成之后释放,且必须保证retain和release的次数对等

那么ARC 实际上帮我们做了哪些事?

ARC不是垃圾回收,也并不是不需要内存管理了,它是隐式的内存管理,编译器在编译的时候会在代码插入合适的ratain和release语句,相当于在背后帮我们完成了内存管理的工作。

什么时候需要自己手动创建autorelease pool?

1)你写的程序不是基于UI framework, 例如命令行项目

2)你写的循环创建了大量临时对象 -> 你需要在循环体内创建一个autorelease pool block并且在每次循环结束之前处理那些autoreleased对象. 在循环中使用autorelease pool block可以降低内存峰值

3)你创建了一个新线程,当线程开始执行的时候你必须立马创建一个autorelease pool , 否则你的应用会造成内存泄露.

相关文章

  • iOS夯实:ARC时代的内存管理

    iOS夯实:ARC时代的内存管理 iOS夯实:ARC时代的内存管理

  • iOS-Swift-内存管理

    一. 关于内存管理 跟OC一样,Swift也是采取基于引用计数的ARC内存管理方案ARC的引用计数管理一般是针对堆...

  • 关于ARC内存管理

    ARC是编译器(时)特性,而不是运行时特性,更不是垃圾回收器(GC)。 OC是使用引用计数机制(retainCou...

  • 关于ARC内存管理

    前言:这段时间我四处投简历面试,然后发现很多以前的基础知识有点记不清了,所以趁这段时间赶紧恶补一下有关方面。 序言...

  • OC中内存管理

    在OC中内存管理MRC手动内存管理和ARC自动内存管理,ARC是从iOS 4.0开始,在iOS 4.0之前...

  • 内存管理

    ARC内存管理机制详解理解 iOS 的内存管理

  • iOS 面试注意事项

    对mrc和arc的理解:OC知识--彻底理解内存管理(MRC、ARC) - 简书 谈谈对自动释放池的理解:关于自动...

  • 内存及性能优化

    1. 用ARC管理内存 ARC除了帮你避免内存泄露,ARC还可以帮你提高性能,它能保证释放掉不再需要的对象的内存。...

  • iOS开发之Autoreleasepool简介

    Autoreleasepool即自动释放池,是在ARC自动管理内存机制下用来管理程序中开辟的内存的,ARC工程每个...

  • OC - OC的内存管理机制

    导读 一、为什么要进行内存管理 二、内存管理机制 三、内存管理原则 四、MRC手动内存管理 五、ARC自动内存管理...

网友评论

      本文标题:关于ARC内存管理

      本文链接:https://www.haomeiwen.com/subject/xsfgsftx.html