nr
Obraz jeśli odpowiedź brzmi tak, to gdzie NSInvocation
jest wystarczająco inteligentny, aby skopiować blok, powinien zrobić coś takiego:
for (/*every arguments*/) {
if (/*arg is object. i.e. @encode(arg) is '@'*/) {
if ([arg isKindOfClss:[NSBlock class]]) {
arg = [arg copy]; // copy block
} else {
[arg retain];
}
}
}
problemem jest to, że arg
jest modyfikowany podczas kopiowania blok, co nie powinno się zdarzyć, ponieważ oznacza to, że wywołanie retainArguments
może zmienić argumenty w NSInvocation
. to złamie wiele założeń, które już zostały wprowadzone. (Tj argumenty dostać od NSInvocation
powinny być takie same jak argumenty używane do tworzenia NSInvocation
) Aktualizacja
właśnie zrobiłem test w celu dostosowania odpowiedź jest NO, ale mój poprzedni punkt był niepoprawny chociaż ...
@interface Test : NSObject
@end
@implementation Test
- (void)testMethodWithBlock:(void (^)(void))block obj:(id)obj cstr:(const char *)cstr {
NSLog(@"%p %p %p %@", block, obj, cstr, [block class]);
}
@end
@implementation testTests
- (void)test1 {
__block int dummy;
Test *t = [[Test alloc] init];
NSMethodSignature *ms = [t methodSignatureForSelector:@selector(testMethodWithBlock:obj:cstr:)];
NSInvocation *invocation = [NSInvocation invocationWithMethodSignature:ms];
void (^block)(void) =^{
dummy++; // stop this become global block
};
id obj = @"object";
char *cstr = malloc(5);
strcpy(cstr, "cstr");
NSLog(@"%@", [ms debugDescription]);
NSLog(@"%p %p %p %@", block, obj, cstr, [block class]);
[invocation setSelector:@selector(testMethodWithBlock:obj:cstr:)];
[invocation setArgument:&block atIndex:2];
[invocation setArgument:&obj atIndex:3];
[invocation setArgument:&cstr atIndex:4];
[invocation invokeWithTarget:t];
[invocation retainArguments];
[invocation invokeWithTarget:t];
free(cstr);
}
@end
wyjście, ARC wyłączony (i awarii):
2013-04-18 19:49:27.616 test[94555:c07] 0xbfffe120 0x70d2254 0x7167980 __NSStackBlock__
2013-04-18 19:49:27.617 test[94555:c07] 0xbfffe120 0x70d2254 0x7167980 __NSStackBlock__
2013-04-18 19:49:27.618 test[94555:c07] 0xbfffe120 0x70d2254 0x736a810 __NSStackBlock__
ARC włączona:
2013-04-18 19:51:03.979 test[95323:c07] 0x7101e10 0x70d2268 0x7101aa0 __NSMallocBlock__
2013-04-18 19:51:03.979 test[95323:c07] 0x7101e10 0x70d2268 0x7101aa0 __NSMallocBlock__
2013-04-18 19:51:03.980 test[95323:c07] 0x7101e10 0x70d2268 0xe0c1310 __NSMallocBlock__
jak widać, c ciąg są kopiowane przez retainArguments
ale nie bloki. ale przy włączonym ARC problem powinien zniknąć, ponieważ ARC skopiował go w pewnym momencie.
Dzięki za aktualizację! – matt