id 和 instancetype

1.动态数据类型

id是一个数据类型, 并且是一个动态数据类型

  • 动态数据类型的特点:

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

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

    id obj = [Person new];
    [obj sleep];
    [obj test];
    [obj eat];
    
    id obj2 = [Student new];
    [obj2 eat];
    [obj2 test];
  • 为了避免动态数据类型引发的运行时的错误, 一般情况下如果使用动态数据类型定义一个变量, 在调用这个对象的方法之前会进行一次判断, 判断当前对象是否能够调用这个方法
    id obj = [Person new];
    id obj = [Student new];
   
    if ([obj isKindOfClass:[Student class]]) {
        // isKindOfClass , 判断指定的对象是否是某一个类, 或者是某一个类的子类
        [obj eat];
    }
  
    if ([obj isMemberOfClass:[Student class]]) {
        // isMemberOfClass : 判断指定的对象是否是当前指定的类的实例
        [obj eat];
    }

2.id 和 instancetype的区别

  • id在编译的时候不能判断对象的真实类型
    instancetype在编译的时候可以判断对象的真实类型
  • id可以用来定义变量, 可以作为返回值, 可以作为形参
    instancetype只能用于作为返回值

注意: 以后但凡自定义构造方法, 返回值尽量使用instancetype, 不要使用id

你可能感兴趣的:(id 和 instancetype)