เป็นไปได้ไหมที่จะส่งผ่านบล็อก Objective-C สำหรับอาร์กิวเมนต์ @selector ใน UIButton
จากคำตอบที่ให้ไว้ทั้งหมดคำตอบคือใช่ แต่จำเป็นต้องใช้งานเล็กน้อยในการตั้งค่าบางหมวดหมู่
ฉันขอแนะนำให้ใช้ NSInvocation เพราะคุณสามารถทำสิ่งนี้ได้มากมายเช่นด้วยตัวจับเวลาเก็บไว้เป็นวัตถุและเรียกใช้ ... ฯลฯ ...
นี่คือสิ่งที่ฉันทำ แต่โปรดทราบว่าฉันใช้ ARC
อันดับแรกเป็นหมวดหมู่ง่ายๆใน NSObject:
.h
@interface NSObject (CategoryNSObject)
- (void) associateValue:(id)value withKey:(NSString *)aKey;
- (id) associatedValueForKey:(NSString *)aKey;
@end
.m
#import "Categories.h"
#import <objc/runtime.h>
@implementation NSObject (CategoryNSObject)
#pragma mark Associated Methods:
- (void) associateValue:(id)value withKey:(NSString *)aKey {
objc_setAssociatedObject( self, (__bridge void *)aKey, value, OBJC_ASSOCIATION_RETAIN );
}
- (id) associatedValueForKey:(NSString *)aKey {
return objc_getAssociatedObject( self, (__bridge void *)aKey );
}
@end
ถัดไปเป็นหมวดหมู่ใน NSInvocation เพื่อจัดเก็บในบล็อก:
.h
@interface NSInvocation (CategoryNSInvocation)
+ (NSInvocation *) invocationWithTarget:(id)aTarget block:(void (^)(id target))block;
+ (NSInvocation *) invocationWithSelector:(SEL)aSelector forTarget:(id)aTarget;
+ (NSInvocation *) invocationWithSelector:(SEL)aSelector andObject:(__autoreleasing id)anObject forTarget:(id)aTarget;
@end
.m
#import "Categories.h"
typedef void (^BlockInvocationBlock)(id target);
#pragma mark - Private Interface:
@interface BlockInvocation : NSObject
@property (readwrite, nonatomic, copy) BlockInvocationBlock block;
@end
#pragma mark - Invocation Container:
@implementation BlockInvocation
@synthesize block;
- (id) initWithBlock:(BlockInvocationBlock)aBlock {
if ( (self = [super init]) ) {
self.block = aBlock;
} return self;
}
+ (BlockInvocation *) invocationWithBlock:(BlockInvocationBlock)aBlock {
return [[self alloc] initWithBlock:aBlock];
}
- (void) performWithTarget:(id)aTarget {
self.block(aTarget);
}
@end
#pragma mark Implementation:
@implementation NSInvocation (CategoryNSInvocation)
#pragma mark - Class Methods:
+ (NSInvocation *) invocationWithTarget:(id)aTarget block:(void (^)(id target))block {
BlockInvocation *blockInvocation = [BlockInvocation invocationWithBlock:block];
NSInvocation *invocation = [NSInvocation invocationWithSelector:@selector(performWithTarget:) andObject:aTarget forTarget:blockInvocation];
[invocation associateValue:blockInvocation withKey:@"BlockInvocation"];
return invocation;
}
+ (NSInvocation *) invocationWithSelector:(SEL)aSelector forTarget:(id)aTarget {
NSMethodSignature *aSignature = [aTarget methodSignatureForSelector:aSelector];
NSInvocation *aInvocation = [NSInvocation invocationWithMethodSignature:aSignature];
[aInvocation setTarget:aTarget];
[aInvocation setSelector:aSelector];
return aInvocation;
}
+ (NSInvocation *) invocationWithSelector:(SEL)aSelector andObject:(__autoreleasing id)anObject forTarget:(id)aTarget {
NSInvocation *aInvocation = [NSInvocation invocationWithSelector:aSelector
forTarget:aTarget];
[aInvocation setArgument:&anObject atIndex:2];
return aInvocation;
}
@end
นี่คือวิธีการใช้งาน:
NSInvocation *invocation = [NSInvocation invocationWithTarget:self block:^(id target) {
NSLog(@"TEST");
}];
[invocation invoke];
คุณสามารถทำอะไรได้มากมายด้วยการเรียกใช้และวิธี Objective-C มาตรฐาน ตัวอย่างเช่นคุณสามารถใช้ NSInvocationOperation (initWithInvocation :), NSTimer (scheduleTimerWithTimeInterval: invocation: repeat :)
ประเด็นคือการเปลี่ยนบล็อกของคุณให้เป็น NSInvocation มีความหลากหลายมากขึ้นและสามารถใช้งานได้ดังนี้:
NSInvocation *invocation = [NSInvocation invocationWithTarget:self block:^(id target) {
NSLog(@"My Block code here");
}];
[button addTarget:invocation
action:@selector(invoke)
forControlEvents:UIControlEventTouchUpInside];
นี่เป็นเพียงคำแนะนำเดียว
objc_implementationWithBlock()
และclass_addMethod()
แก้ไขปัญหานี้ได้อย่างมีประสิทธิภาพมากกว่าการใช้วัตถุที่เกี่ยวข้องเล็กน้อย (ซึ่งหมายความว่าการค้นหาแฮชที่ไม่มีประสิทธิภาพเท่ากับการค้นหาวิธีการ) น่าจะเป็นความแตกต่างของประสิทธิภาพที่ไม่เกี่ยวข้อง แต่ก็เป็นอีกทางเลือกหนึ่ง