Block-Closures和三元运算符,用于使用动态代码枚举NSArray

时间:2012-05-23 12:01:02

标签: objective-c objective-c-blocks dynamic-code

我不知道我想做的事情是否可行。我对块没有经验,只是阅读了一些教程,并且非常喜欢这个想法。 我的类AppointmentView.h(和m)正在扩展一个ViewController。它包含一些属性,这些属性是另一个Class Appointment的成员,后者又有一些NSArrays。 为了创建分组的表视图,我检查数组的值,然后适当地复制将要显示的值(没有太多花哨或复杂)。 通常情况下,我会使用if-else语句并适当地控制,但是在查看块后,它让我想到以下是否可行:

[self setOutcomes: [[NSArray alloc] initWithObjects:
                        ^{ return [myAppointment.CANCELED isEqualToString:@"NO"] == YES ? 
                                                            [[NSString alloc] initWithString:@"Not Cancelled"] :
                                                            [[NSString alloc] initWithString:@"Cancelled"]; },
                        ^{ return [myAppointment.CANCELED isEqualToString:@"NO"] == YES ?
                                                            [[NSString alloc] initWithString:@"No Cancellation reason"] :
                                                            [[NSString alloc] initWithString:myAppointment.CANCREASON]; },
                        ^{ return [myAppointment EVENTS].length > 0 ?
                                                            [[NSString alloc] initWithString:myAppointment.EVENTS] :
                                                            [[NSString alloc] initWithString:@"No Events"]; },
                        ^{ return [myAppointment SUMMARY].length > 0 ?
                            [[NSString alloc] initWithString:myAppointment.SUMMARY] :
                            [[NSString alloc] initWithString:@"No Summary"]; },
                        nil]];

此代码编译良好,但运行时崩溃。我的理解是这是错误的,因为我告诉枚举每次执行一段代码。所以这不是我想要的。 然后我尝试了以下内容:

   [self setOutcomes: [[NSArray alloc] initWithObjects:
                        [[NSString alloc] initWithString: (NSString *) ^(void){ 
                                                                        return [myAppointment.CANCELED isEqualToString:@"NO"] == YES ?
                                                                                                        @"Not Cancelled" : @"Cancelled"; }],
                        [[NSString alloc] initWithString: (NSString *) ^(void){ 
                                                                        return [myAppointment.CANCELED isEqualToString:@"NO"] == YES ? 
                                                                                                        @"No Cancellation reason" : myAppointment.CANCREASON; }],
                        [[NSString alloc] initWithString: (NSString *) ^(void){ 
                                                                        return [myAppointment.EVENTS isEqualToString:@""] == YES ? 
                                                                                                        myAppointment.EVENTS : @"No Events"; }],
                        [[NSString alloc] initWithString: (NSString *) ^(void){ 
                                                                        return [myAppointment.SUMMARY isEqualToString:@""] == YES ?
                                                                                                        myAppointment.SUMMARY : @"No Summary"; }],
                        nil]];

虽然它编译得很好,但崩溃时出现以下错误:

  

- [ NSMallocBlock 长度]:无法识别的选择器发送到实例0x6bd68f0

我也尝试过定义一个我可以使用的通用方法,但也没有帮助。 所以,我想知道,这可能吗?我知道对于这段代码,我可以简单地使用if-else(就像我通常那样)并完成它,但是在其他情况下,这样的动态代码生成选项将非常有用。

我觉得NSString不知道如何处理这个问题,因此我应该用一个选项来扩展它,让我们说

  

initWithBlock:

或者我在这里错过了其他的东西。或者它可能是不可能的....(?)

PS:我刚刚看过NSArray的enumerateUsingBlock:方法,但我不确定在这种情况下这是否符合我的目的,因为数组中的每个条目都完全不同于另一个,并且特定的块不适合标准。

2 个答案:

答案 0 :(得分:3)

如果您需要在初始化程序中做出简单的选择,则不需要块。

[NSArray initWithObjects:
         ([myAppointment.CANCELED isEqualToString:@"NO"] ? @"Not Cancelled" : @"Cancelled"),
         ([myAppointment.CANCELED isEqualToString:@"NO"] ? @"No Cancellation reason" : myAppointment.CANCREASON),
         nil];

使用parens括起?:运算符。


更新以回答有关如何在同一位置定义和调用块的问题

这个单元测试对我有用:

- (void)testBlockNow
{
    BOOL X = ^{ return YES; }();
    STAssertTrue(X, nil);
}

这很简单:^{ … }定义块,然后()调用块。

答案 1 :(得分:2)

是的,你可以在一个数组中存储块,但你需要聪明一点。

typedef (void)(^X)();

X block1 = ^{ … };
X block2 = ^{ … };
X block3 = ^{ … };

NSArray *array = [NSArray initWithObjects:
                  [[block1 copy] autorelease],
                  [[block2 copy] autorelease],
                  [[block3 copy] autorelease],
                  nil];