Always pass weak reference of self into block in ARC? Always pass weak reference of self into block in ARC? ios ios

Always pass weak reference of self into block in ARC?


It helps not to focus on the strong or weak part of the discussion. Instead focus on the cycle part.

A retain cycle is a loop that happens when Object A retains Object B, and Object B retains Object A. In that situation, if either object is released:

  • Object A won't be deallocated because Object B holds a reference to it.
  • But Object B won't ever be deallocated as long as Object A has a reference to it.
  • But Object A will never be deallocated because Object B holds a reference to it.
  • ad infinitum

Thus, those two objects will just hang around in memory for the life of the program even though they should, if everything were working properly, be deallocated.

So, what we're worried about is retain cycles, and there's nothing about blocks in and of themselves that create these cycles. This isn't a problem, for example:

[myArray enumerateObjectsUsingBlock:^(id obj, NSUInteger idx, BOOL *stop){   [self doSomethingWithObject:obj];}];

The block retains self, but self doesn't retain the block. If one or the other is released, no cycle is created and everything gets deallocated as it should.

Where you get into trouble is something like:

//In the interface:@property (strong) void(^myBlock)(id obj, NSUInteger idx, BOOL *stop);//In the implementation:[self setMyBlock:^(id obj, NSUInteger idx, BOOL *stop) {  [self doSomethingWithObj:obj];     }];

Now, your object (self) has an explicit strong reference to the block. And the block has an implicit strong reference to self. That's a cycle, and now neither object will be deallocated properly.

Because, in a situation like this, self by definition already has a strong reference to the block, it's usually easiest to resolve by making an explicitly weak reference to self for the block to use:

__weak MyObject *weakSelf = self;[self setMyBlock:^(id obj, NSUInteger idx, BOOL *stop) {  [weakSelf doSomethingWithObj:obj];     }];

But this should not be the default pattern you follow when dealing with blocks that call self! This should only be used to break what would otherwise be a retain cycle between self and the block. If you were to adopt this pattern everywhere, you'd run the risk of passing a block to something that got executed after self was deallocated.

//SUSPICIOUS EXAMPLE:__weak MyObject *weakSelf = self;[[SomeOtherObject alloc] initWithCompletion:^{  //By the time this gets called, "weakSelf" might be nil because it's not retained!  [weakSelf doSomething];}];


I totally agree with @jemmons:

But this should not be the default pattern you follow when dealing with blocks that call self! This should only be used to break what would otherwise be a retain cycle between self and the block. If you were to adopt this pattern everywhere, you'd run the risk of passing a block to something that got executed after self was deallocated.

//SUSPICIOUS EXAMPLE:__weak MyObject *weakSelf = self;[[SomeOtherObject alloc] initWithCompletion:^{  //By the time this gets called, "weakSelf" might be nil because it's not  retained!  [weakSelf doSomething];}];

To overcome this problem one can define a strong reference over the weakSelf inside the block:

__weak MyObject *weakSelf = self;[[SomeOtherObject alloc] initWithCompletion:^{  MyObject *strongSelf = weakSelf;  [strongSelf doSomething];}];


You don't have to always use a weak reference. If your block is not retained, but executed and then discarded, you can capture self strongly, as it will not create a retain cycle. In some cases, you even want the block to hold the self until the completion of the block so it does not deallocate prematurely. If, however, you capture the block strongly, and inside capture self, it will create a retain cycle.