较短的回答(Short Answer)
这是一个历史原因,在ARC中不使用Copy,而使用Strong是完全正确的。就像是使用实例变量的时候,是使用local还是global。
较长的回答(Long Answer)
和其他的object不一样,一个block是被存储在stack上的,这是一个优化的实现,这种实现是很必要的。像其他的编译优化一样,这对code没有一个直接的影响。这个优化在block被创建,作为一个方法的参数被传入再被释放的时候受益于一个一般的案例。block在stack上被创建,释放的时候也不需要heap(动态内存池)的调用。
把它和局部变量做对比,(a)一个是在stack上创建,(b)在方法return的时候自动被摧毁,(c)在被调用的时候,可以以地址的方式传入到方法中。在方法return的时候,局部变量的地址不能够被存储和使用——已经被释放了。
但是,如果需要,对象的预期持续时间可以比创建的方法长久。所以不像局部变量,它们被创建在heap上并且在方法创建的时候不会自动释放,而是取决于局部变量是否被需要,是否“需要”是由ARC决定的。
在stack上创建block能得到一些的好处,但是也会引起一个问题。如果block想要持续存在,就像object做的,所以在创建它的stack释放之前它必须要移动到heap上。
当实现的block被第一个释放,在stack上strong block的优化是可见的对于开发者,编译器在这个时候不能够自动处理移动block到heap上。当需要的时候,开发者必须自己调用block_copy
方法。
但是这个方法只能在底层C的API中使用(block是一个C的结构体),在Object-C的层开发的开发者手动管理编译优化不是一个很好的实践。Apple发布的最新的编译器版本改善了这种使用,较早的开发者被告诉用[block copy]
替代block_copy(block)
,这符合一般的Object-C的调用方式。当需要的时候,编译器开始自动的从stack拷贝出block。但是这没有被官方文档正式说明。
没有必要暂时的手动从stack拷贝block,因为Apple不能摆脱历史的包袱,只能认为这样做是best practice,但这也是相当有争议的。在2014年9月最新的版本中,苹果的Workding with Blocks陈述了block属性应该使用copy
,这就立马变的清晰了。
Note:你应该用copy来修饰block属性。为了追踪原始作用域捕获的状态,block需要被拷贝。当使用Automatic Reference Counting你就不需要担心这些,因为这会自动发生,但是属性attribute命名的最佳实践应该是能够说明它的必然行为。
没有必要显示“它必然的行为” - 在stack上的第一个地方存储block是一个优化,并且对于code来说应该是透明可见的。就像其他编译器优化之后,不应该需要开发者做什么就能得到性能的提升。
个人理解的意思是:编译器优化了之后,不应该要求使用者将属性改为copy,使用者只要知道被strong过的block也会想普通对象一样被ARC管理就行了。
所以只要你使用ARC,现在的Clang编译器,你可以像对待其他对象一样对待block。因为block是可变的。这意味着你不需要拷贝他们。相信Apple,他们已经在编译器优化的时候帮我们做了这件事。他们鼓励你抛弃历史的包袱,copy
不再需要了。
有很多地方没有翻译通,但是为什么用strong代替copy已经很清楚了:
之前我们需要手动的用copy
属性来修饰block,让block从stack拷贝到heap,保证block在出了作用域之后也能够让block继续存在,并且以ARC的方式来决定什么时候释放在heap上的block。在2014年9月后的一次编译器优化之后,如果用strong
修饰block,编译器会自动将blcok从stack拷贝到heap上。
对于开发者来说,这就像处理普通对象一样用strong
来block就行了。
原文:Cocoa blocks as strong pointers vs copy
欢迎指正👏👏👏👏
Object C中的对象存放在Heap上还是Stack上
Object-C只使用heap对象,不使用stack对象。
Stack
stack包含了局部变量的存储的原始内存,每个执行的thread都有一个stack。当一个方法被called,方法内的局部变量的数据存储在stack地址。当方法return的时候,stack的地址被释放。所有的这些都会自动发生。
Heap
内存中的一部分是heap。在heap的上的内存任何时候都能被分配和摧毁。
所以,最后stack上的object的的内存地址是在heap被分配的。
原文:where does Objective C store objects, heap or stack [duplicate]
欢迎指正👏👏👏👏
网友评论