instancetype和id异同

本文详细介绍了Objective-C中的instancetype和id关键字的区别与用法,包括它们在关联返回类型和非关联返回类型方法中的作用,以及如何通过instancetype提高代码的可读性和错误预防能力。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

一、什么是instancetype

instancetype是clang 3.5开始,clang提供的一个关键字,表示某个方法返回的未知类型的Objective-C对象。我们都知道未知类型的的对象可以用id关键字表示,那为什么还会再有一个instancetype呢?

二、关联返回类型(related result types)

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

1、类方法中,以alloc或new开头

2、实例方法中,以autorelease,init,retain或self开头

会返回一个方法所在类类型的对象,这些方法就被称为是关联返回类型的方法。换句话说,这些方法的返回结果以方法所在的类为类型,说的有点绕口,请看下面的例子:

[objc]  view plain copy 在CODE上查看代码片 派生到我的代码片
  1. @interface NSObject  
  2. + (id)alloc;  
  3. - (id)init;  
  4. @end  
  5.   
  6. @interface NSArray : NSObject  
  7. @end  
当我们使用如下方式初始化NSArray时:
[objc]  view plain copy 在CODE上查看代码片 派生到我的代码片
  1. NSArray *array = [[NSArray alloc] init];  
按照Cocoa的命名规则,语句 [NSArray alloc]  的类型就是NSArray* 因为alloc的返回类型属于关联返回类型。同样,[[NSArray alloc]init]  的返回结果也是 NSArray*

三、instancetype作用

1、作用

如果一个不是关联返回类型的方法,如下:

[objc]  view plain copy 在CODE上查看代码片 派生到我的代码片
  1. @interface NSArray  
  2. + (id)constructAnArray;  
  3. @end  

当我们使用如下方式初始化NSArray时:

[objc]  view plain copy 在CODE上查看代码片 派生到我的代码片
  1. [NSArray constructAnArray];  
根据Cocoa的方法命名规范, 得到的返回类型就和方法声明的返回类型一样,是id

但是如果使用instancetype作为返回类型,如下:

[objc]  view plain copy 在CODE上查看代码片 派生到我的代码片
  1. @interface NSArray  
  2. + (instancetype)constructAnArray;  
  3. @end  
当使用相同方式初始化NSArray时:
[objc]  view plain copy 在CODE上查看代码片 派生到我的代码片
  1. [NSArray constructAnArray];  
得到的返回类型和方法所在类的类型相同, NSArray* !

总结一下,instancetype的作用,就是使那些非关联返回类型的方法返回所在类的类型

2、好处

能够确定对象的类型,能够帮助编译器更好的为我们定位代码书写问题,比如:

[objc]  view plain copy 在CODE上查看代码片 派生到我的代码片
  1. [[[NSArray alloc] init] mediaPlaybackAllowsAirPlay]; //  "No visible @interface for `NSArray` declares the selector `mediaPlaybackAllowsAirPlay`"  
  2.   
  3. [[NSArray array] mediaPlaybackAllowsAirPlay]; // (No error)  
上例中第一行代码,由于 [[NSArray   alloc] init] 的结果是 NSArray* ,这样编译器就能够根据返回的数据类型检测出NSArray是否实现mediaPlaybackAllowsAirPlay方法。有利于开发者在编译阶段发现错误。

第二行代码,由于array不属于关联返回类型方法,[NSArray array]返回的是id类型,编译器不知道id类型的对象是否实现了mediaPlaybackAllowsAirPlay方法,也就不能够替开发者及时发现错误。

四、instancetype和id的异同

1、相同点

都可以作为方法的返回类型

2、不同点

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

②instancetype只能作为返回值,不能像id那样作为参数,比如下面的写法:

[objc]  view plain copy 在CODE上查看代码片 派生到我的代码片
  1. //err,expected a type  
  2. - (void)setValue:(instancetype)value  
  3. {  
  4.     //do something  
  5. }  
就是错的,应该写成:
[objc]  view plain copy 在CODE上查看代码片 派生到我的代码片
  1. - (void)setValue:(id)value  
  2. {  
  3.     //do something  
  4. }  


### iOS Objective-C `instancetype` 关键字详解 #### 定义与作用 在Objective-C中,`instancetype` 是一种特殊的返回类型修饰符。当方法可能返回当前类实例或其子类实例时,使用此关键字有助于编译器更精确地推断对象的实际类型[^1]。 #### 使用场景 主要应用于工厂模式下的初始化方法以及便利构造函数中。例如创建数组、字符串等常用数据类型的实例化操作: ```objective-c @interface NSArray (FactoryMethods) // 返回值为 id 类型的方法声明 + (id)array; // 返回值为 instancetype 的方法声明 + (instancetype)arrayWithCapacity:(NSUInteger)numItems; @end ``` 采用 `instancetype` 可使代码更加安全可靠,在调用者看来返回的就是所期望的具体类型而不是泛化的 `id` 或父类指针形式[^4]。 #### 实际案例分析 考虑一个简单的自定义视图控制器基类及其派生类的情况: ```objective-c #import <UIKit/UIKit.h> @interface BaseViewController : UIViewController + (instancetype)viewControllerWithTitle:(NSString *)title; @end @implementation BaseViewController + (instancetype)viewControllerWithTitle:(NSString *)title { BaseViewController *vc = [[self alloc] init]; vc.title = title; return vc; // 编译期即能确定返回的是 BaseViewController 或其子类的对象 } @end @interface ChildViewController : BaseViewController @end ChildViewController *childVC = [ChildViewController viewControllerWithTitle:@"Child"]; [childVC setTitle:@"New Title"]; // 不需要强制转换即可访问属性 ``` 上述例子展示了如何利用 `instancetype` 来提高代码可读性减少不必要的类型转换错误风险[^2]。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值