Block concept

  • Blocks are anonymous functions with automatic variables. As it stands, blocks have no function name. In addition, blocks have an insertion symbol “^”, which makes it easier to find the Block, followed by a parenthesis that represents a list of arguments that the Block needs. Like a function, you can pass arguments to a block and also have a return value.

  • The difference is that a block is defined inside a function or method and can access any variable within the scope of the function or method. Normally, these variables can be accessed but cannot change their value, and a special block modifier (consisting of two underscore characters before the block) can change the value of variables within the block.

  • Blocks are also called closures, code Blocks. A block is a code block that encapsulates the code you want to execute and calls it when needed.

Let’s start with a simple Block

int main(int argc, const char * argv[]) {
    @autoreleasepool {
        void(^block)(void) = ^{
            NSLog(@"B123");
        };
        block();
 }
    return 0;
}
Copy the code

Use the command line to convert the code to c++ to see its internal structure

xcrun -sdk iphoneos clang -arch arm64 -rewrite-objc main.m
Copy the code

The C++ code is as follows

struct __block_impl { void *isa; int Flags; int Reserved; void *FuncPtr; }; struct __main_block_impl_0 { struct __block_impl impl; struct __main_block_desc_0* Desc; Constructor (similar to OC's init method), Struct __main_block_impl_0(void *fp, struct __main_block_desc_0 *desc, int flags=0) { impl.isa = &_NSConcreteStackBlock; impl.Flags = flags; impl.FuncPtr = fp; //block code address Desc = Desc; // Store the size of the block object}}; Static void __main_block_func_0(struct __main_block_impl_0 *__cself) {NSLog((NSString) *)&__NSConstantStringImpl__var_folders_3f_7jst2vt56bd_nhbrl1hlv5s00000gn_T_main_e9907b_mi_0); } static struct __main_block_desc_0 { size_t reserved; size_t Block_size; } __main_block_desc_0_DATA = { 0, sizeof(struct __main_block_impl_0)}; int main(int argc, const char * argv[]) { /* @autoreleasepool */ { __AtAutoreleasePool __autoreleasepool; / / block variables and void block (*) (void) = ((void (*) ()) & __main_block_impl_0 (__main_block_func_0 (void *), &__main_block_desc_0_DATA)); / / implementation code within the block (void (__block_impl *) (*)) ((__block_impl *) block) - > FuncPtr) ((__block_impl *) block); } return 0; }Copy the code
Void (^ Block)(void) = ^{NSLog(@"B123"); }; The corresponding C++ code is:  void(*block)(void) = ((void (*)())&__main_block_impl_0((void *)__main_block_func_0, &__main_block_desc_0_DATA));Copy the code

Block is called __main_block_IMPL_0, which is implemented as follows:

struct __main_block_impl_0 { struct __block_impl impl; struct __main_block_desc_0* Desc; Constructor (similar to OC's init method), Struct __main_block_impl_0(void *fp, struct __main_block_desc_0 *desc, int flags=0) { impl.isa = &_NSConcreteStackBlock; impl.Flags = flags; impl.FuncPtr = fp; //block code address Desc = Desc; // Store the size of the block object}};Copy the code

__main_block_IMPL_0 (void *fp, struct __main_block_desC_0 * DESC, int flags=0); This is the C++ constructor, which is equivalent to the OC class – (instancetype)init; Methods. The __main_block_IMPL_0 function takes three arguments

1. void *fp 
2. struct __main_block_desc_0 *desc
3.  int flags=0
Copy the code

__main_block_desc_0 is the structure pointer to which the function pointer is referred. __main_block_desc_0 is the structure pointer to which the function pointer is referred. In the demo’s C++ implementation, the function fp points to is __main_block_func_0, and __main_block_func_0 is the code that executes inside the block, encapsulated in this function:

static void __main_block_func_0(struct __main_block_impl_0 *__cself) {

         NSLog((NSString *)&__NSConstantStringImpl__var_folders_3f_7jst2vt56bd_nhbrl1hlv5s00000gn_T_main_e9907b_mi_0);
        }
Copy the code

Struct __main_block_desc_0 *desc

static struct __main_block_desc_0 { size_t reserved; // 0 size_t Block_size; } __main_block_desc_0_DATA = {0, sizeof(struct __main_block_impl_0)};Copy the code

We in the execution of void (block) (void) = ((void () ()) & __main_block_impl_0 ((void *) __main_block_func_0, & __main_block_desc_0_DATA)); Pass in the address of this function and assign it to the internal variable of the structure. The last argument is optional, and the default value is 0. This tells us that a Block is essentially a structure, so let’s look inside that structure

struct __block_impl {
  void *isa;
  int Flags;
  int Reserved;
  void *FuncPtr;
};


struct __main_block_impl_0 {
  struct __block_impl impl;
  struct __main_block_desc_0* Desc;
Copy the code

Next, compare the calling code

OC
 block();

C++
 ((void (*)(__block_impl *))((__block_impl *)block)->FuncPtr)((__block_impl *)block);
Copy the code

We can see that the call to Block is cast to impl and then FuncPtr

A more complicated Block

int main(int argc, const char * argv[]) {
    @autoreleasepool {
        int age;
        age = 30;
      void(^block)(void) = ^(){
                  NSLog(@"%d",age);
              };
        age = 40
        block();

    }

    return 0;
}
Copy the code

If you’ve ever used a Block, you’re probably familiar with it, but if you print a knot of 30, let’s take a look at the underlying implementation

struct __main_block_impl_0 { struct __block_impl impl; struct __main_block_desc_0* Desc; int age; }; struct __main_block_impl_0 { struct __block_impl impl; struct __main_block_desc_0* Desc; int age; __main_block_impl_0(void *fp, struct __main_block_desc_0 *desc, int _age, int flags=0) : age(_age) { impl.isa = &_NSConcreteStackBlock; impl.Flags = flags; impl.FuncPtr = fp; Desc = desc; }}; static void __main_block_func_0(struct __main_block_impl_0 *__cself) { int age = __cself->age; // bound by copy NSLog((NSString *)&__NSConstantStringImpl__var_folders_3f_7jst2vt56bd_nhbrl1hlv5s00000gn_T_main_440a9c_mi_0,age); }Copy the code

You can see that an age variable is generated inside the Block to store it, and the value of the building Block is passed inside, so its value does not change

It is now also clear that the inner structure of the underlying Block is as follows

capture

concept

In order to ensure normal access to external variables within a block, a variable Capture mechanism is provided to Capture variables within the scope of the block definition, similar to saving the context. Later, when a Block is called elsewhere, it executes as if it were still in the original context. These are all things that OC does automatically. And when we define a Block, any variables that are currently in scope can be used in the Block. Don’t care if the Block is called somewhere else

We define three types of variables

  1. The global variable
  2. A local variable
  3. A static variable
int globalint ; int main(int argc, const char * argv[]) { @autoreleasepool { auto int stakeint = 10; static int staticint = 20; void(^block)(void) = ^(){ NSLog(@"%d%d%d",globalint,stakeint,staticint); }; // void(^block)(int,int) = ^(int a,int b){ // NSLog(@"%d",a + b); / /}; struct __main_block_impl_0 * blockstruct = (__bridge struct __main_block_impl_0*)block; block(); } C++ //Block structure declaration int globalint; struct __main_block_impl_0 { struct __block_impl impl; struct __main_block_desc_0* Desc; int stakeint; int *staticint; __main_block_impl_0(void *fp, struct __main_block_desc_0 *desc, int _stakeint, int *_staticint, int flags=0) : stakeint(_stakeint), staticint(_staticint) { impl.isa = &_NSConcreteStackBlock; impl.Flags = flags; impl.FuncPtr = fp; Desc = desc; }}; Static void __main_block_func_0(struct __main_block_impl_0 *__cself) {int case = __cself-> Case; static void __main_block_func_0(struct __main_block_impl_0 *__cself) {int case = __cself-> case; // bound by copy int *staticint = __cself->staticint; // bound by copy NSLog((NSString *)&__NSConstantStringImpl__var_folders_3f_7jst2vt56bd_nhbrl1hlv5s00000gn_T_main_faa523_mi_0,globalint,stakeint,(*statici nt)); }Copy the code

Summarize the schematic

When we determine whether a Block captures an object we need to determine whether it is a global variable, if not, and if not

If you look at the underlying implementation of the Block type, you’ll see the following statement

impl.isa = &_NSConcreteStackBlock;
Copy the code

So this is essentially saying the object of a Block

We declare one of these blocks, and then print the parent to the root class

  void(^blockTest)(void) = ^(){
                         NSLog(@"Test");
                     };
        
        NSLog(@"%@",[blockTest class]);
        //NSLog(@"%@",[[blockTest class] superclass]);
        NSLog(@"%@",[blockTest superclass]);
                  
        NSLog(@"%@",[[blockTest superclass]superclass]);
        NSLog(@"%@",[[[blockTest superclass]superclass]superclass]);
Copy the code

The print result is as follows

We can see that the last root class is NSObject, so we can also say that the Block is an OC object

The type,

There are three types of blocks, which you can see by calling the class method or isa pointer. They are all derived from the NSBlock type

  • NSGlobalBlock (_NSConcreteGlobalBlock)
  • NSStackBlock (_NSConcreteStackBlock)
  • NSMallocBlock (_NSConcreteMallocBlock)

Layout of blocks in memory

Block type assignment

🍪

oid (^Block)(void);
void test3() {
    int number = 10;
    Block = ^{
        NSLog(@"%d", number);
    };
}
int main(int argc, const char * argv[]) {
    @autoreleasepool {
        test3();
        Block();               
    }
    return 0;
}
Copy the code

ARC and MRC will have different results. If you know about memory management, you should know that ARC is a compiler feature. That is, the compiler helps us to do a lot of things and simplify our code

Second, we can see that the Block is of type

And the reason why you’re printing an error is very simple because the memory of your previous Block structure is on the Stack, and the test method is recycled, so printing an error is also very simple to solve the problem, let’s just store the Block on the heap, so according to the previous flowchart, As soon as we do a copy operation we can first look at the Block type and then print 10

So we’re looking at the second test

void test3() { int number = 10; Block = [^{ NSLog(@"%d", number); } copy]; } int main(int argc, const char * argv[]) { @autoreleasepool { void (^BlockTest)(void)= [^{ NSLog(@"123"); } copy]; NSLog(@"%@", [BlockTest class]); } test3(); [Block copy]; NSLog(@"%@", [Block class]); [Block release]; return 0; } // BlockTest is __NSGlobalBlock__ // print the result __NSGlobalBlock__ // __NSMallocBlock__Copy the code

The copy of the ARC

In an ARC environment, the compiler automatically copies blocks on the stack to the heap as needed, such as the following

  • Block as a function return value
  • Assigns a block to a __strong pointer
  • Block is used as a Cocoa API method name containing a method parameter called usingBlock
  • Block as a method parameter of the GCD API

To summarize

Each type of block calls copy as shown below

The auto variable of object type

An example leads to this

 {
        Person * person = [[Person alloc] init];
        person.age =10;
        }
        NSLog(@"1111111");
Copy the code
   Block block;
        {
        Person * person = [[Person alloc] init];
        person.age =10;
           block  = ^{
                NSLog(@"%d",person.age);
            };
        
        }
        NSLog(@"1111111");
Copy the code
Print the result // the first object will be released until 1111111 is printed // the second object will not be released until 1111111 is printedCopy the code

So let’s move on to C++ code and take a look at the underlying implementation

struct __main_block_impl_0 { struct __block_impl impl; struct __main_block_desc_0* Desc; Person *person; __main_block_impl_0(void *fp, struct __main_block_desc_0 *desc, Person *_person, int flags=0) : person(_person) { impl.isa = &_NSConcreteStackBlock; impl.Flags = flags; impl.FuncPtr = fp; Desc = desc; }}; static void __main_block_func_0(struct __main_block_impl_0 *__cself) { Person *person = __cself->person; // bound by copy NSLog((NSString *)&__NSConstantStringImpl__var_folders_3f_7jst2vt56bd_nhbrl1hlv5s00000gn_T_main_00319d_mi_0,((int (*)(id, SEL))(void *)objc_msgSend)((id)person, sel_registerName("age"))); }Copy the code

Person *person = __cself-> Person, which points the person inside the Block to the person outside the instance object. Since the external instance object is held by the block, it will not be freed. When the block is freed, the Person will be freed

Block block; { Person * person = [[Person alloc] init]; person.age =10; __weak Person * weakPerson = person; block = ^{ NSLog(@"%d",weakPerson.age); }; } NSLog(@"1111111"); Struct __main_block_impl_0 {struct __block_impl impl; struct __main_block_desc_0* Desc; Person *__weak weakPerson; __main_block_impl_0(void *fp, struct __main_block_desc_0 *desc, Person *__weak _weakPerson, int flags=0) : weakPerson(_weakPerson) { impl.isa = &_NSConcreteStackBlock; impl.Flags = flags; impl.FuncPtr = fp; Desc = desc; }}; static void __main_block_func_0(struct __main_block_impl_0 *__cself) { Person *__weak weakPerson = __cself->weakPerson; // bound by copy NSLog((NSString *)&__NSConstantStringImpl__var_folders_3f_7jst2vt56bd_nhbrl1hlv5s00000gn_T_main_9af25a_mi_0,((int (*)(id, SEL))(void *)objc_msgSend)((id)weakPerson, sel_registerName("age"))); } // We can see that the underlying person modifier is weakCopy the code
Static void __main_block_copy_0(struct __main_block_impl_0* DST, struct __main_block_impl_0) struct __main_block_impl_0*src) {_Block_object_assign((void*)&dst->weakPerson, (void*)src->weakPerson, 3/*BLOCK_FIELD_IS_OBJECT*/); } static void __main_block_dispose_0(struct __main_block_impl_0*src) {_Block_object_dispose((void*)src->weakPerson, 3/*BLOCK_FIELD_IS_OBJECT*/); } static struct __main_block_desc_0 { size_t reserved; size_t Block_size; void (*copy)(struct __main_block_impl_0*, struct __main_block_impl_0*); void (*dispose)(struct __main_block_impl_0*); } __main_block_desc_0_DATA = { 0, sizeof(struct __main_block_impl_0), __main_block_copy_0, __main_block_dispose_0};Copy the code

To summarize

When a block accesses the auto variable of object type internally if the block is on the stack, There will be no strong reference to the auto variable and the copy function inside the block will be called if the block is copied to the heap The _Block_object_assign function is called internally. The block_object_assign function performs operations based on the __strong, __weak, and __unsafe_unretained parameter of the auto variable. Dispose function is called inside the block if it is removed from the heap

The _Block_object_dispose function is called internally and automatically releases the referenced auto variable (release).

At this point we can take a look at the internal structure of the complete Block

__block

role

__block can be used to solve the problem of internal block cannot modify auto variable value. __block cannot modify global variable, static variable (static)

typedef void(^Block)(void);
int main(int argc, const char * argv[]) {
    @autoreleasepool {
        __block int  age = 5 ;
        Block block = ^{
            age = 10;
            NSLog(@"%d",age);
        };
        block();
    }
Copy the code

So let’s go ahead and convert to C++ code

typedef void(*Block)(void); struct __Block_byref_age_0 { void *__isa; __Block_byref_age_0 *__forwarding; int __flags; int __size; int age; }; struct __main_block_impl_0 { struct __block_impl impl; struct __main_block_desc_0* Desc; __Block_byref_age_0 *age; // by ref __main_block_impl_0(void *fp, struct __main_block_desc_0 *desc, __Block_byref_age_0 *_age, int flags=0) : age(_age->__forwarding) { impl.isa = &_NSConcreteStackBlock; impl.Flags = flags; impl.FuncPtr = fp; Desc = desc; }}; static void __main_block_func_0(struct __main_block_impl_0 *__cself) { __Block_byref_age_0 *age = __cself->age; // bound by ref (age- g->age) = 10; NSLog((NSString *)&__NSConstantStringImpl__var_folders_3f_7jst2vt56bd_nhbrl1hlv5s00000gn_T_main_526302_mi_0,(age->__forwarding->age)); } static void __main_block_copy_0(struct __main_block_impl_0*dst, struct __main_block_impl_0*src) {_Block_object_assign((void*)&dst->age, (void*)src->age, 8/*BLOCK_FIELD_IS_BYREF*/); } static void __main_block_dispose_0(struct __main_block_impl_0*src) {_Block_object_dispose((void*)src->age, 8/*BLOCK_FIELD_IS_BYREF*/); } static struct __main_block_desc_0 { size_t reserved; size_t Block_size; void (*copy)(struct __main_block_impl_0*, struct __main_block_impl_0*); void (*dispose)(struct __main_block_impl_0*); } __main_block_desc_0_DATA = { 0, sizeof(struct __main_block_impl_0), __main_block_copy_0, __main_block_dispose_0}; // __block int age = 5 ; __attribute__((__blocks__(byref))) __Block_byref_age_0 age = {(void*)0, (__Block_byref_age_0 *)&age, 0, sizeof(__Block_byref_age_0), 5 }; Block block = ((void (*)())&__main_block_impl_0((void *)__main_block_func_0, &__main_block_desc_0_DATA, (__Block_byref_age_0 *)&age, 570425344)); ((void (*)(__block_impl *))((__block_impl *)block)->FuncPtr)((__block_impl *)block);Copy the code

We can see that the compiler wraps the __block variable as an object, which can be used to solve the problem of the block not being able to modify the value of the auto variable

⚠️ so what if we print age here?

__block int  age = 5 ;
    Block block = ^{
        age = 10;
        NSLog(@"%d",age);
    };
    block();
    struct __main_block_impl_0 * blockLook = (__bridge struct __main_block_impl_0*)block;
    NSLog(@"%p",&age);
Copy the code

And here we can see that it is

(__Block_byref_age_0 *) age = 0x00000001005526b0
Copy the code

But our printed results were off by 24, and when we look back at the structure the answer should be obvious

That’s the age inside, so when we change it, it’s the encapsulated value

Memory management of __block

When a block is on the stack, There is no strong reference to the __block variable. When a block is copied to the heap, the block’s internal copy function is called, which calls the _Block_object_assign function The _Block_object_assign function forms a strong reference to the __block variable (retain)

The following figure

When a block is removed from the heap, its internal dispose function is called, which calls the _Block_object_dispose function The _Block_object_dispose function automatically releases the referenced __block variable (release)

__block decorates an object

Let’s start with a simple example

//OC __block Person * person = [[Person alloc] init]; MyBlock block = ^{ NSLog(@"%p", person); }; C++ struct __block_impl { void *isa; int Flags; int Reserved; void *FuncPtr; }; struct __Block_byref_person_0 { void *__isa; __Block_byref_person_0 *__forwarding; int __flags; int __size; void (*__Block_byref_id_object_copy)(void*, void*); void (*__Block_byref_id_object_dispose)(void*); Person *person; }; struct __main_block_impl_0 { struct __block_impl impl; struct __main_block_desc_0* Desc; __Block_byref_person_0 *person; // by ref __main_block_impl_0(void *fp, struct __main_block_desc_0 *desc, __Block_byref_person_0 *_person, int flags=0) : person(_person->__forwarding) { impl.isa = &_NSConcreteStackBlock; impl.Flags = flags; impl.FuncPtr = fp; Desc = desc; }}; static void __main_block_func_0(struct __main_block_impl_0 *__cself) { __Block_byref_person_0 *person = __cself->person;  // bound by ref NSLog((NSString *)&__NSConstantStringImpl__var_folders_3f_7jst2vt56bd_nhbrl1hlv5s00000gn_T_main_fd7a72_mi_0, (person->__forwarding->person)); } static void __main_block_copy_0(struct __main_block_impl_0*dst, struct __main_block_impl_0*src) {_Block_object_assign((void*)&dst->person, (void*)src->person, 8/*BLOCK_FIELD_IS_BYREF*/); } static void __main_block_dispose_0(struct __main_block_impl_0*src) {_Block_object_dispose((void*)src->person, 8/*BLOCK_FIELD_IS_BYREF*/); } static struct __main_block_desc_0 { size_t reserved; size_t Block_size; void (*copy)(struct __main_block_impl_0*, struct __main_block_impl_0*); void (*dispose)(struct __main_block_impl_0*); }Copy the code

We can see its memory structure as shown below

If the pointer is strong or weak, we can use __weak

Person * person = [[Person alloc] init]; __block __weak Person * weakperson = person; MyBlock block = ^{ NSLog(@"%p", weakperson); }; struct __Block_byref_weakperson_0 { void *__isa; __Block_byref_weakperson_0 *__forwarding; int __flags; int __size; void (*__Block_byref_id_object_copy)(void*, void*); void (*__Block_byref_id_object_dispose)(void*); Person *__weak weakperson; }; struct __main_block_impl_0 { struct __block_impl impl; struct __main_block_desc_0* Desc; __Block_byref_weakperson_0 *weakperson; // by ref __main_block_impl_0(void *fp, struct __main_block_desc_0 *desc, __Block_byref_weakperson_0 *_weakperson, int flags=0) : weakperson(_weakperson->__forwarding) { impl.isa = &_NSConcreteStackBlock; impl.Flags = flags; impl.FuncPtr = fp; Desc = desc; }};Copy the code

Knowing that __block objects are always strong Pointers, but internal Pointers are strong or weak depending on your modifier, let’s take a look at the internal structure of __Block_byref_weakperson_

struct __Block_byref_weakperson_0 {
  void *__isa;
__Block_byref_weakperson_0 *__forwarding;
 int __flags;
 int __size;
 void (*__Block_byref_id_object_copy)(void*, void*);
 void (*__Block_byref_id_object_dispose)(void*);
 Person *__weak weakperson;
};
Copy the code

And found it inside

void (*__Block_byref_id_object_copy)(void*, void*);
 void (*__Block_byref_id_object_dispose)(void*);
Copy the code

This is also reference technology management for its objects

To summarize the

The auto and __block variables of the object type

The same

When blocks are on the stack, there are no strong references to any of them

When blocks are copied to the heap, they are processed by the copy function

A __block variable (assuming the variable name is a)

_Block_object_assign((void*)&dst->a, (void*)src->a, 8/*BLOCK_FIELD_IS_BYREF*/);
Copy the code

The auto variable of the object type (assuming the variable name is p)

_Block_object_assign((void*)&dst->p, (void*)src->p, 3/*BLOCK_FIELD_IS_OBJECT*/);
Copy the code

Whenever blocks are removed from the heap, dispose of their __block variables (assuming the variable name is a) via the dispose function

_Block_object_dispose((void*)src->a, 8/*BLOCK_FIELD_IS_BYREF*/);
Copy the code

The auto variable of the object type (assuming the variable name is p)

_Block_object_dispose((void*)src->p, 3/*BLOCK_FIELD_IS_OBJECT*/);
Copy the code

__blocks at different points hold their objects

__forwarding pointer to __block

(age->__forwarding->age) = 10;

The reason for this is shown in the figure below, so that we can access __block decorated variables on both the stack and the queue

_forwarding points to itself. In block construction, we use A. __forwarding, and in use (__BlockTest__test_block_func_0) we also find the final value ‘8’ by (a->__forwarding->a). The reason for this is described above, and repeated here: copying a __block variable on the stack to the heap replaces the value of the member variable __forwarding with the address of the struct instance of the __block variable copied to the heap. Therefore, “the __block variable can be accessed correctly regardless of whether it is configured on the stack or heap”, which is also the reason for the existence of the member variable __forwarding. Refer to the diagram above

A circular reference to a Block

Block the essence

  • A block is essentially an object, and that object isa structure that contains an isa pointer to its own class.
  • A block is an OC object that encapsulates a function call and its environment
  • A block is an OC object that encapsulates a function and its context.

See objecPron -C for advanced programming


Click:For more information

Citation: Link to original text