NSLog(@"%@",[NSMutableArray arrayWithObject:@""].class); __NSArrayM框架
NSLog(@"%@",@[@"",@""].class); __NSArrayI指针
NSLog(@"%@",@[].class); __NSArray0code
当咱们建立一个NSArray对象时,实际上获得的是NSArray的子类__NSArrayI
对象.一样的,咱们建立NSMutableArray
对象,获得的一样是其子类__NSArrayM
对象.有趣的是,当咱们建立只有一个对象的NSArray
时,获得的是__NSSingleObjectArrayI
类对象.__NSArrayI
和__NSArrayM
,__NSSingleObjectArrayI
为框架隐藏的类.对象
经过NSArray和NSMutableArray接口,返回的倒是子类对象,怎么作到的?
先介绍另外一个私有类:__NSPlaceholderArray
,和两个此类的全局变量___immutablePlaceholderArray
,___mutablePlaceholderArray
。__NSPlaceholderArray
从类命名上看,它只是用来占位的,具体怎么占位法稍后讨论,有个重要特色是,__NSPlaceholderArray
实现了和NSArray
,NSMutableArray
一摸同样的初始化方法,如initWithObjects:count:
,initWithCapacity:
等.接口
介绍完__NSPlaceholderArray
后,这个机制能够总结为如下两个大步骤:
(1).NSArray重写了+ (id)allocWithZone:(struct _NSZone *)zone
方法,在方法内部,若是调用类为NSArray
则直接返回全局变量___immutablePlaceholderArray
,若是调用类为NSMUtableArray
则直接返回全局变量___mutablePlaceholderArray
。
也就是调用[NSArray alloc]
或者[NSMUtableArray alloc]
获得的仅仅是两个占位指针,类型为__NSPlaceholderArray
.
(2).在调用了alloc
的基础上,不管是NSArray
或NSMutableArray
都一定要继续调用某个initXXX
方法,而实际上调用的是__NSPlaceholderArray
的initXXX
.在这个initXXX
方法内部,若是self == ___immutablePlaceholderArray
就会从新构造并返回__NSArrayI
对象,若是self == ___mutablePlaceholderArray
就会从新构造并返回_NSArrayM
对象.ci
总结来讲,对于NSArray
和NSMutableArray
,alloc
时拿到的仅仅是个占位对象,init
后才获得真实的子类对象.it