6

考虑:

@interface Foo : NSObject

+ (void) dump ;

@end

@implementation Foo

+ (Class) classOf1 {
    return self ;
}

+ (Class) classOf2 {
    return [Foo class] ;
}

+ (Class) classOf3 {
    return [[[Foo class] new] class] ;
}

+ (Class) classOf4 {
    return [[self new] class] ;
}

+ (Class) classOf5 {
    return [[[self alloc] init] class] ;
}

+ (Class) classOf6 {
    return [[[Foo alloc] init] class] ;
}

+ (Class) classOf7 {
    return [self class] ;
}

+ (void) dump {
    NSLog(@"classOf1 %@<0x%08.8lx>", [self classOf1], (unsigned long)[[self classOf1] hash]) ;
    NSLog(@"classOf2 %@<0x%08.8lx>", [self classOf2], (unsigned long)[[self classOf2] hash]) ;
    NSLog(@"classOf3 %@<0x%08.8lx>", [self classOf3], (unsigned long)[[self classOf3] hash]) ;
    NSLog(@"classOf4 %@<0x%08.8lx>", [self classOf4], (unsigned long)[[self classOf4] hash]) ;
    NSLog(@"classOf5 %@<0x%08.8lx>", [self classOf5], (unsigned long)[[self classOf5] hash]) ;
    NSLog(@"classOf6 %@<0x%08.8lx>", [self classOf6], (unsigned long)[[self classOf6] hash]) ;
    NSLog(@"classOf7 %@<0x%08.8lx>", [self classOf7], (unsigned long)[[self classOf7] hash]) ;
}

@end

和输出:

    2013-07-04 03:20:20.404 WC[29862:c07] classOf1 Foo<0x0002a2e4>
    2013-07-04 03:20:21.075 WC[29862:c07] classOf2 Foo<0x0002a2e4>
    2013-07-04 03:20:21.628 WC[29862:c07] classOf3 Foo<0x0002a2e4>
    2013-07-04 03:20:22.229 WC[29862:c07] classOf4 Foo<0x0002a2e4>
    2013-07-04 03:20:22.805 WC[29862:c07] classOf5 Foo<0x0002a2e4>
    2013-07-04 03:20:23.387 WC[29862:c07] classOf6 Foo<0x0002a2e4>
    2013-07-04 03:20:25.235 WC[29862:c07] classOf7 Foo<0x0002a2e4>

所有 7 个案例都返回完全相同的值!

对我来说最令人费解的案例:

  • classOf1 和 classOf7

在后者中,“自我”“代表” Foo 这是一个类。那么一堂课是什么课呢?

  • classOf5 和 classOf6

显然,“Foo”和“self”在这种情况下扮演着完全相同的角色。正如他们在

  • classOf3 和 classOf4

在我看来,“Foo”有点像魔法兔子。

你可以

@class Foo ;

在某种程度上

@class self ;

没有意义。

在词法分析器到运行时生成器的管道中,我可以理解比特流何时从一堆字符变为一个identifier依赖于语言的语义元素。但是我很难弄清楚 ObjC 类标识符的确切性质。

任何人?

(这个SO Question似乎无关)

4

2 回答 2

8

您可以在NSObject 的实现中看到答案:

+ (Class)class {
    return self;
}

- (Class)class {
    return object_getClass(self);
}

发送class到类对象不会返回其元类,只返回类对象本身。

如果出于某种原因需要元类对象,则必须使用运行时函数object_getClass(),如下所示:

NSLog(@"%d", class_isMetaClass(object_getClass([NSString class])));
// Prints 1

您还可以使用.objc_getMetaClass()

于 2013-07-04T03:08:28.613 回答
5
+ (Class) classOf1 {
    return self ;
}

self在一个类方法中是Class当前类的对象,所以这里self是类的类对象Foo


+ (Class) classOf2 {
    return [Foo class] ;
}

这是类的类对象Foo


+ (Class) classOf3 {
    return [[[Foo class] new] class] ;
}

这和

Class fooClass = [Foo class];
id fooInstance = [fooClass new];
return [fooInstance class];

再一次,给你类的类对象Foo


+ (Class) classOf4 {
    return [[self new] class] ;
}

与上一个相同,只需替换[Foo class]self


+ (Class) classOf5 {
    return [[[self alloc] init] class] ;
}

与上一个相同,只需替换[Foo new][[Foo alloc] init]


+ (Class) classOf6 {
    return [[[Foo alloc] init] class] ;
}

与上一个相同,只需替换selfFoo


+ (Class) classOf7 {
    return [self class] ;
}

与第二个相同,只需替换Fooself


Foo在类方法中使用和的区别在于self,当在子类上调用此方法时,self将被替换为子类。例如

@interface Bar : Foo
@end

Class barClass = [Bar classOf1]; // same as [Bar class]
Class fooClass = [Bar classOf2]; // same as [Foo class]

如果你真的想得到类的类(即元类),你需要使用来自的objc运行时函数<objc/runtime.h>

id metaClass = objc_getMetaClass("Foo");
于 2013-07-04T03:08:48.207 回答