Objective-C 中 id, instancetype, NSObject* 区别

1,122 阅读4分钟

1. id

idobjc.h中定义

struct objc_object {
    Class _Nonnull isa;
};
typedef struct objc_object *id;

id的本质是objc_object结构体指针,idOC对象,可以指向任何OC对象。

2. idNSObject *区别

  1. id是OC 对象,但是并不一定是NSObject对象。OC中的基类除了NSObject之外,还有一个NSProxy虚类。因此,id相比NSObject *指向的对象范围要更广。所以id可以强转为其他类型, NSObject 只能强转为其子类。
  2. id动态数据类型,而NSObject *静态数据类型,默认情况下所有的数据类型都是静态
  3. id指针不能使用点语法。

静态数据类型的特点:

  • 在编译时就知道变量的类型, 以及变量中有哪些属性和方法,编译时就可以访问这些属性和方法;
  • 通过静态数据类型定义变量, 如果访问了不属于该数据类型的属性和方法, 那么编译器就会报错。

动态数据类型的特点:

  • 在编译的时编译器并不知道变量的真实类型, 只有在运行的时候才知道它的真实类型
  • 通过动态数据类型定义变量, 如果访问了不属于动态数据类型的属性和方法, 编译器不会报错 id弊端: 由于动态数据类型可以调用任意方法, 所以有可能调用到不属于自己的方法, 而编译时又不会报错, 所以可能导致运行时的错误

id应用场景: 多态, 可以减少代码量, 避免调用子类特有的方法需要强制类型转换。

为了避免动态数据类型引发的运行时的错误, 一般情况下如果使用动态数据类型定义一个变量, 在调用该对象的方法之前会进行一次判断, 判断当前对象是否能够调用该方法 。

3. instancetype简介

1. 什么是instancetype

instancetypeclang 3.5开始,clang提供的一个关键字,表示某个方法返回的未知类型的Objective-C对象。

我们都知道未知类型的的对象可以用id关键字表示,那为什么还会再有一个instancetype呢?

2. 关联返回类型

根据Cocoa的命名规则,满足下述规则的方法:

  1. 类方法中,allocnew开头。
  2. 实例方法中,以autoreleaseinitretainself开头。

会返回一个方法所在类类型的对象,这些方法就被称为是关联返回类型的方法。换言之,这些方法的返回值类型是该方法所在的类为类型,如下所示:

@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. idinstancetype区别

instancetype只能作为返回值。id 除了作为返回值,还可以定义变量、作为参数;

instancetype可以返回和方法所在类相同类型的对象,id只能返回未知类型的对象。

instancetype在编译的时候可以判断对象的真实类型,id在编译的时候不能判断对象的真实类型。