Executing Blocks From NSArray?
Solution 1
Sure, you just invoke it with ()
like any other block, but you need to typecast the value you retrieve from NSArray
. Here's an example (with an added typedef, because otherwise my head hurts):
typedef int (^IntBlock)(void);
IntBlock Block_001 = ^{ return 101; };
IntBlock Block_002 = ^{ return 202; };
NSArray *array = [NSArray arrayWithObjects:Block_001, Block_002, nil];
int x = ((IntBlock)[array objectAtIndex:0]) (); // now x == 101
Solution 2
@KennyTM and @David are correct, but your code is potentially wrong. Here's why:
When creating an NSArray
with objects, it will retain
the objects put into it. In the case of blocks, it's using the Block_retain
function. This means that the array has retained the blocks that you created, but that live on the stack (blocks are one of the very rare examples of Objective-C objects that can be created on the stack without delving into absurd tricks). That means that as soon as this method exits, your array now points to garbage, because the blocks it was pointing to no longer exist. To do this properly, you should do:
int (^Block_001)(void) = [^{ return 101; } copy];
int (^Block_002)(void) = [^{ return 202; } copy];
NSArray *array = [NSArray arrayWithObjects:Block_001, Block_002, nil];
[Block_001 release];
[Block_002 release];
By invoking copy
on the block, you're explicitly moving the block off of the stack and onto the heap, where it can safely remain after the method/function exits. Then after you've added the blocks to the array, you have to balance your copy
(because of the NARC rule) with a subsequent call to release
. Make sense?
Solution 3
Of course you can.
int (^x)(void) = [array objectAtIndex:0];
printf("%d\n", x()); // prints 101.
fuzzygoat
Apple Development @ Fuzzygoat, A digital nomad adventuring in one possible future.
Updated on June 03, 2022Comments
-
fuzzygoat almost 2 years
I was just thinking, as you can treat Blocks like objects if I create two of them and then add them to an NSArray is there a way to execute them from the array?
int (^Block_001)(void) = ^{ return 101; }; int (^Block_002)(void) = ^{ return 202; }; NSArray *array = [NSArray arrayWithObjects:Block_001, Block_002, nil];
EDIT: Update for clarity Per @davedelong's excellent answer
int (^Block_001)(void) = [^{ return 101; } copy]; int (^Block_002)(void) = [^{ return 202; } copy]; NSArray *array = [NSArray arrayWithObjects:Block_001, Block_002, nil]; [Block_001 release]; [Block_002 release];
-
fuzzygoat almost 14 yearsHi Dave, yes I understand, I was not aware of blocks being created on the stack (only just started looking at blocks today). Much appreciated.
-
fuzzygoat almost 14 yearsok, so not being familiar with function pointers am I getting this right? The typedef is defining "IntBlock" that is a pointer to a block that returns an int and takes no arguments. I think I get it, and looking at the alternative (which I am pretty sure I would get wrong) I appreciate your decision to go with the typedef :) Much appreciated.
-
fuzzygoat almost 14 yearsI did consider this, very interesting. Can you just clarify for me, am I right in referring to "int (^x)(void)" as a pointer to a block? Just trying to make sure I am getting the terminology correct.
-
aroth over 12 yearsYou don't need to cast it. It works just fine to do
int (^block)() = [array objectAtIndex:0]
. A pointer is a pointer is a pointer, after all, at least in low-level languages likeC
and its derivatives. -
David Gelhar over 12 years@aroth Sure, you can also use an assignment to a variable of the appropriate type to perform the "conversion" of the generic
id
type returned from-objectAtIndex
. What you can't do isint y = [array objectAtIndex:0]();
(the compiler will complain "called typeid
is not a function or function pointer"). One way (cast) or another (assignment to a block pointer), you need to tell the compiler "I know this thing I'm pulling out of the array is a block, so when I say()
, please call it!" -
Tieme over 11 yearsHi Dave, I found this question and David Gelhar's accepted answer so I implemented it and it worked quite well, it didn't crash. Reading further, your answer made sense but does not explain why david's code is not crashing (immediately?). Do you have any clue?
-
Dave DeLong over 11 years@Tieme David's code doesn't crash immediately because when you execute the block stored in the array, you're still in the same lexical scope in which the block was created. If you were to return that array as the return value of the method, and then try and execute the blocks in the calling function, then it would crash. Although if you're using ARC it probably won't, because the compiler will insert the necessary copy call for you.
-
Tieme over 11 yearsOkay got, it, I'll test it. But although the compiler will probably insert the copy you shouldn't leave it out like
release
andretain
right?