看一下NSArray.h中的NSArray创建方法块。
返回id的方法不返回instancetype是否有正当的理由?
苹果甚至努力添加内联注释,以使我们知道在这种情况下id返回一个NSArray。

@interface NSArray (NSArrayCreation)

+ (instancetype)array;
+ (instancetype)arrayWithObject:(id)anObject;
+ (instancetype)arrayWithObjects:(const id [])objects count:(NSUInteger)cnt;
+ (instancetype)arrayWithObjects:(id)firstObj, ... NS_REQUIRES_NIL_TERMINATION;
+ (instancetype)arrayWithArray:(NSArray *)array;

- (instancetype)init;   /* designated initializer */
- (instancetype)initWithObjects:(const id [])objects count:(NSUInteger)cnt; /* designated   initializer */

- (instancetype)initWithObjects:(id)firstObj, ... NS_REQUIRES_NIL_TERMINATION;
- (instancetype)initWithArray:(NSArray *)array;
- (instancetype)initWithArray:(NSArray *)array copyItems:(BOOL)flag;

+ (id /* NSArray * */)arrayWithContentsOfFile:(NSString *)path;
+ (id /* NSArray * */)arrayWithContentsOfURL:(NSURL *)url;
- (id /* NSArray * */)initWithContentsOfFile:(NSString *)path;
- (id /* NSArray * */)initWithContentsOfURL:(NSURL *)url;

@end
我唯一想出的这些特殊方法是Apple的指导

但是,对于我来说,这仍然不能解释在instancetype上使用id的原因,因为它们仍然允许NSArray子类返回自己的instancetypeNSDictionary遵循完全相同的模式,其中使用文件或URL的内容创建字典使用id,所有其他创建方法使用instancetype
- (instancetype)initWithObjectsAndKeys:(id)firstObject, ... NS_REQUIRES_NIL_TERMINATION;
- (instancetype)initWithDictionary:(NSDictionary *)otherDictionary;
- (instancetype)initWithDictionary:(NSDictionary *)otherDictionary copyItems:(BOOL)flag;
- (instancetype)initWithObjects:(NSArray *)objects forKeys:(NSArray *)keys;

+ (id /* NSDictionary * */)dictionaryWithContentsOfFile:(NSString *)path;
+ (id /* NSDictionary * */)dictionaryWithContentsOfURL:(NSURL *)url;
- (id /* NSDictionary * */)initWithContentsOfFile:(NSString *)path;
- (id /* NSDictionary * */)initWithContentsOfURL:(NSURL *)url;
我知道Apple只是打算将基础类中的id替换为instancetype,但是在单个类中使用时出现的模式不一致会指导我们自己使用,还是只是绕开他们开始工作的类而已上?
扩展一下,我想在调用NSMutableDictionary时探索dictionaryWithContentsOfFile的返回类型
NSString * plistPath = [[NSBundle mainBundle] pathForResource:@"myFile" ofType:@"plist"];
NSMutableDictionary *myDictionary = [NSMutableDictionary dictionaryWithContentsOfFile:plistPath];
    if ([ myDictionary isKindOfClass:[NSMutableDictionary class]])
    {
        NSLog(@"This is a mutable dictionary why id and not instancetype?");
        [myDictionary setObject:@"I can mutate the dictionary" forKey:@"newKey"];
    }
NSLog (@"%@", myDictionary[@"newKey"]);
    return YES;
}
以下输出到我的控制台:

因此,我能够向字典中添加新的键和对象。

最佳答案

类群集可能会返回您从其创建类之外的其他类。
对于Foundation类,这通常是正确的,因为在许多情况下,它们将创建某种优化的类。该类仍将从isKindOfClass返回YES:

同样,一些免费的桥接类返回的是在Foundation和Core Foundation之间共享的类。一个示例是NSCFString。

09-25 16:25