使用类方法的Objective-C点符号?

时间:2010-03-03 23:49:08

标签: objective-c

注意,我特别指的是点符号与类方法一起使用,而不是实例方法。

出于好奇,我想看看如果我尝试将Objective-C点符号语法与类方法一起使用会发生什么。我的实验如下:

#import <Foundation/Foundation.h>

static int _value = 8;

@interface Test : NSObject

+ (int) value;
+ (void) setValue:(int)value;

@end

@implementation Test

+ (int) value {
    return _value;
}

+ (void) setValue:(int)value {
    _value = value;
}

@end

int main(int argc, char * argv[]) {

    NSAutoreleasePool * pool = [[NSAutoreleasePool alloc] init];

    NSLog(@"Test.value: %d", Test.value);
    NSLog(@"[Test value]: %d", [Test value]);

    Test.value = 20;
    NSLog(@"Test.value: %d", Test.value);
    NSLog(@"[Test value]: %d", [Test value]);

    [Test setValue:30];
    NSLog(@"Test.value: %d", Test.value);
    NSLog(@"[Test value]: %d", [Test value]);

    [pool release];

    return 0;
}

我很惊讶地看到这是编译的,更不用说我认为正确的行为了。这是在某处记录的,还是只是编译器的侥幸?

我在Mac OS X 10.6上使用GCC编译:

gcc --version: i686-apple-darwin10-gcc-4.2.1 (GCC) 4.2.1 (Apple Inc. build 5659) 

compile using: gcc ObjCClassDotSyntax.m -framework Foundation -o ObjCClassDotSyntax
run: ./ObjCClassDotSyntax

output:
2010-03-03 17:33:07.342 test[33368:903] Test.value: 8
2010-03-03 17:33:07.346 test[33368:903] [Test value]: 8
2010-03-03 17:33:07.351 test[33368:903] Test.value: 20
2010-03-03 17:33:07.352 test[33368:903] [Test value]: 20
2010-03-03 17:33:07.353 test[33368:903] Test.value: 30
2010-03-03 17:33:07.353 test[33368:903] [Test value]: 30

3 个答案:

答案 0 :(得分:32)

这是正确的行为。 foo.method[foo method]的语法糖 - 具有相同语义的直接转换。类似地,foo.prop = bar[foo setProp:bar]的语法糖,同样具有相同的语义。此转换在编译器中实现。因此,可以使用点表示法来调用0参数方法,如foo.doSomething而不是[foo doSomething]。当然,如果你这样做,你就是邪恶

被调用者是一个类实例的事实并不重要,因为在Objective-C中,类也是对象。在类上使用点表示法会在该类上调用无参数方法。

Objective-C Programming Language文档中描述了点符号。

答案 1 :(得分:12)

在“邪恶但它有用”类别中,我已经知道偶尔使用带有点符号的便捷构造函数,例如NSMutableArray *myArray = NSMutableArray.array

答案 2 :(得分:0)

Underscore库通过从类方法返回块来进一步滥用此语法,从而产生如下代码:

NSArray *elements = Underscore.array(array)
    .flatten
    .uniq
    .unwrap;

要了解其工作原理,请查看Underscore.array的定义:

+ (USArrayWrapper *(^)(NSArray *))array
{
    return ^(NSArray *array) {
        return [USArrayWrapper wrap:array];
    };
}

所以:

Underscore.array(array) 

......等同于:

NSArray *array = @[];
USArrayWrapper * (^arr)(NSArray *) = [Underscore array];
USArrayWrapper *result = arr(array);