Blocks and Variables(Block和变量)
This article describes the interaction between blocks and variables, including memory management.
本文讲述了block和变量之间的内在关系,包括内存管理。
Types of Variable(变量类型)
Within the block object’s body of code, variables may be treated in five different ways.
You can reference three standard types of variable, just as you would from a function:
-
Global variables, including static locals
-
Global functions (which aren’t technically variables)
-
Local variables and parameters from an enclosing scope
- 全局变量,包括静态的本地变量
- 全局函数,(这个并不是技术上的变量)
- 本地变量,和同一个作用域的参数
Blocks also support two other types of variable:
-
At function level are
__block
variables. These are mutable within the block (and the enclosing scope) and are preserved if any referencing block is copied to the heap. -
const
imports.
- 在函数级别的_block 变量。这些变量在block中是可变的,同时如果引用变量的block拷贝进堆堆,手动管理内存)内存就会保存 。
- 常量
The following rules apply to variables used within a block:
-
Global variables are accessible, including static variables that exist within the enclosing lexical scope.
-
Parameters passed to the block are accessible (just like parameters to a function).
-
Stack (non-static) variables local to the enclosing lexical scope are captured as
const
variables.Their values are taken at the point of the block expression within the program. In nested blocks, the value is captured from the nearest enclosing scope.
-
Variables local to the enclosing lexical scope declared with the
__block
storage modifier are provided by reference and so are mutable.Any changes are reflected in the enclosing lexical scope, including any other blocks defined within the same enclosing lexical scope. These are discussed in more detail inThe __block Storage Type.
-
Local variables declared within the lexical scope of the block, which behave exactly like local variables in a function.
Each invocation of the block provides a new copy of that variable. These variables can in turn be used as
const
or by-reference variables in blocks enclosed within the block.
- 全局变量可以访问,包括作用域内的静态变量。(这里block有对全局变量的读写权限)
- 传入block中的参数可以访问(就像函数中的参数)。
- 作用域中的栈(自动内存管理)(非静态)变量,被当做常量来使用。 在程序中他们的值被block表达式调用。在嵌套block中,会使用最近的作用域中的变量。(笔者的理解是,嵌套的只能用最近的。PS:这一大段其实就几个字:block中普通变量是只读的)
- 作用域中的_block变量提供的是引用,所以是可变的。 所有的变动都会反映在作用域中,包括同一个作用域内定义了其他的block。更具体的讨论:见The __block Storage Type。
- block中声明的局部变量,他们的行为很像函数中的局部变量。 block的每次调用都会给变量提供一份新的拷贝。这些在block作用域中的变量可以是常量,也可以做变量(引用的)。
The following example illustrates the use of local non-static variables:
下列例子说明了非静态局部变量变量的使用
int x = 123; |
|
void (^printXAndY)(int) = ^(int y) { |
|
printf("%d %d\n", x, y); |
}; |
|
printXAndY(456); // prints: 123 456 |
As noted, trying to assign a new value to x
within the block would result in an error:
值得注意的是,试图在块内给block赋值会返回error
int x = 123; |
|
void (^printXAndY)(int) = ^(int y) { |
|
x = x + y; // error |
printf("%d %d\n", x, y); |
}; |
To allow a variable to be changed within a block, you use the __block
storage type modifier—see The __block Storage Type.
The __block Storage Type(_block 存储类型)
You can specify that an imported variable be mutable—that is, read-write— by applying the __block
storage type modifier. __block
storage is similar to, but mutually exclusive of, the register
, auto
, and static
storage types for local variables.
__block
variables live in storage that is shared between the lexical scope of the variable and all blocks and block copies declared or created within the variable’s lexical scope. Thus, the storage will survive the destruction of the stack frame if any copies of the blocks declared within the frame survive beyond the end of the frame (for example, by being enqueued somewhere for later execution). Multiple blocks in a given lexical scope can simultaneously use a shared variable.
As an optimization, block storage starts out on the stack—just like blocks themselves do. If the block is copied using Block_copy
(or in Objective-C when the block is sent a copy
), variables are copied to the heap. Thus, the address of a __block
variable can change over time.
There are two further restrictions on __block
variables: they cannot be variable length arrays, and cannot be structures that contain C99 variable-length arrays.
The following example illustrates use of a __block
variable:
下面的例子说说明了_block变量的使用。
__block int x = 123; // x lives in block storage |
|
void (^printXAndY)(int) = ^(int y) { |
|
x = x + y; |
printf("%d %d\n", x, y); |
}; |
printXAndY(456); // prints: 579 456 |
// x is now 579 |
The following example shows the interaction of blocks with several types of variables:
这个例子展现了几种类型的block的相互作用
extern NSInteger CounterGlobal; |
static NSInteger CounterStatic; |
|
{ |
NSInteger localCounter = 42; |
__block char localCharacter; |
|
void (^aBlock)(void) = ^(void) { |
++CounterGlobal; |
++CounterStatic; |
CounterGlobal = localCounter; // localCounter fixed at block creation |
localCharacter = 'a'; // sets localCharacter in enclosing scope |
}; |
|
++localCounter; // unseen by the block |
localCharacter = 'b'; |
|
aBlock(); // execute the block |
// localCharacter now 'a' |
} |
Object and Block Variables (对象和block变量)
Blocks provide support for Objective-C and C++ objects, and other blocks, as variables.
block可以作为变量,支持Objective-C和C++对象,其他block。
Objective-C Objects(Objective—C 对象)
When a block is copied, it creates strong references to object variables used within the block. If you use a block within the implementation of a method:
-
If you access an instance variable by reference, a strong reference is made to
self
; -
If you access an instance variable by value, a strong reference is made to the variable.
- 如果你通过引用访问一个实例变量,就对self建立一个强引用。
- 如果你通过值访问了一个实例变量,就对变量建立一个强引用。
The following examples illustrate the two different situations:
下面的例子说明了不同的情况
dispatch_async(queue, ^{ |
// instanceVariable is used by reference, a strong reference is made to self |
doSomethingWithObject(instanceVariable); |
}); |
|
|
id localVariable = instanceVariable; |
dispatch_async(queue, ^{ |
/* |
localVariable is used by value, a strong reference is made to localVariable |
(and not to self). |
*/ |
doSomethingWithObject(localVariable); |
}); |
To override this behavior for a particular object variable, you can mark it with the __block
storage type modifier.
通过重写一个特定对象变量,你可以使用_block存储类型修饰符来标记它。
C++ Objects (C++对象)
In general you can use C++ objects within a block. Within a member function, references to member variables and functions are via an implicitly imported this
pointer and thus appear mutable. There are two considerations that apply if a block is copied:
-
If you have a
__block
storage class for what would have been a stack-based C++ object, then the usualcopy
constructor is used. -
If you use any other C++ stack-based object from within a block, it must have a
const copy
constructor. The C++ object is then copied using that constructor.
- 如果你有一个基于栈的C++对象,里面有_block存储类,这个时候使用通常构造函数。
- 如果你在一个block中使用其他的基于栈的C++对象,就必须使用常拷贝构造函数。C++对象拷贝的时候会用那个构造函数。