1. id
id在objc.h中定义
struct objc_object {
Class _Nonnull isa;
};
typedef struct objc_object *id;
id的本质是objc_object结构体指针,id是OC对象,可以指向任何OC对象。
2. id与NSObject *区别
id是OC 对象,但是并不一定是NSObject对象。OC中的基类除了NSObject之外,还有一个NSProxy虚类。因此,id相比NSObject *指向的对象范围要更广。所以id可以强转为其他类型, NSObject 只能强转为其子类。id是动态数据类型,而NSObject *是静态数据类型,默认情况下所有的数据类型都是静态。id指针不能使用点语法。
静态数据类型的特点:
- 在编译时就知道变量的类型, 以及变量中有哪些属性和方法,编译时就可以访问这些属性和方法;
- 通过静态数据类型定义变量, 如果访问了不属于该数据类型的属性和方法, 那么编译器就会报错。
动态数据类型的特点:
- 在编译的时编译器并不知道变量的真实类型, 只有在运行的时候才知道它的真实类型
- 通过动态数据类型定义变量, 如果访问了不属于动态数据类型的属性和方法, 编译器不会报错
id弊端: 由于动态数据类型可以调用任意方法, 所以有可能调用到不属于自己的方法, 而编译时又不会报错, 所以可能导致运行时的错误
id应用场景: 多态, 可以减少代码量, 避免调用子类特有的方法需要强制类型转换。
为了避免动态数据类型引发的运行时的错误, 一般情况下如果使用动态数据类型定义一个变量, 在调用该对象的方法之前会进行一次判断, 判断当前对象是否能够调用该方法 。
3. instancetype简介
1. 什么是instancetype
instancetype是clang 3.5开始,clang提供的一个关键字,表示某个方法返回的未知类型的Objective-C对象。
我们都知道未知类型的的对象可以用id关键字表示,那为什么还会再有一个instancetype呢?
2. 关联返回类型
根据Cocoa的命名规则,满足下述规则的方法:
- 类方法中,
alloc或new开头。 - 实例方法中,以
autorelease,init,retain或self开头。
会返回一个方法所在类类型的对象,这些方法就被称为是关联返回类型的方法。换言之,这些方法的返回值类型是该方法所在的类为类型,如下所示:
@interface NSObject
+ (id)alloc;
- (id)init;
@end
@interface NSString : NSObject
@end
//当我们使用如下方式初始化NSString时:
//按照Cocoa的命名规则,语句 [NSString alloc] 的类型就是 NSString*
//因为 alloc 的返回类型属于关联返回类型。
//同样,[[NSString alloc]init] 的返回结果也是 NSString* 。
NSString *arr = [[NSString alloc] init];
3. instancetype作用
就是使那些非关联返回类型的方法返回类型为所在类的类型.如果一个不是关联返回类型的方法,如下所示:
@interface NSString
+ (id)createString;
@end
//根据 Cocoa 的方法命名规范, 得到的返回类型就和方法声明的返回类型一样,是id。
id arr = [NSString createString];
使用instancetype作为返回类型,如下:
@interface NSString
+ (instancetype)constructAnString;
@end
//得到的返回类型和方法所在类的类型相同, 是 NSString*
NSString *arr = [NSString createString];
4. 好处
能够确定对象的类型(编译时就能够检测出返回的数据类型),更好的为我们定位代码书写问题。
4. id与instancetype区别
instancetype只能作为返回值。id 除了作为返回值,还可以定义变量、作为参数;
instancetype可以返回和方法所在类相同类型的对象,id只能返回未知类型的对象。
instancetype在编译的时候可以判断对象的真实类型,id在编译的时候不能判断对象的真实类型。