ios学习之Runtime基础概念

Runtime 又叫运行时,是一套底层的 C 语言 API,其为 iOS 内部的核心之一,我们平时编写的 OC 代码,底层都是基于它来实现的。比如方法调用:

[objA  someMethod];本质上是objc_msgSend(objA,someMethod).

我们常说Objective-C是动态语言,这其实正是得益于Runtime。有了Runtime,我们可以在程序运行时获取某个类的所有成员变量以及方法列表,还可以动态添加方法、方法交换等。

要全面了解OC的Runtime,必须知道以下术语:

SEL、IMP、id、Class、Method、Ivar.

SEL

正如上面所言,在OC当中方法调用实际上是发消息,参数当中的someMethod就是一个SEL类型,是selector在Objc当中的表示,其数据结构是:

typedef  struct objc_selector *SEL;

可以看到,它是一个指向 objc_selector 指针,表示方法的名字/签名(本质就是一个字符串)。所以,对于不同类的的相同的selector,其SEL是一样的。

IMP

实际上,有了SEL还是不能真正的发起方法调用,真正起作用的还是这个IMP。先看一下IMP的数据结构:

IMP在objc.h中的定义是:typedef id(*IMP)(id, SEL, ...);

可以看到IMP实际上是一函数指针,指向什么地方呢?指向这个IMP对应的函数代码在内存中的位置。从结构体可以看出,这里面的第一个参数是id类型,代表要执行该方法的对象;第二个参数,SEL就很好理解了,就是方法的名字。

id

这个术语在上面已经见到过,是指向某个类的实例的指针,数据结构如下:

typedef struct objc_object *id;

struct objc_object { Class isa; };

以上定义,看到objc_object结构体包含一个 isa 指针,根据 isa 指针就可以找到对象所属的类。

注意:

isa 指针在代码运行时并不总指向实例对象所属的类型,所以不能依靠它来确定类型,要想确定类型还是需要用对象的-class方法。KVO 的实现机理就是将被观察对象的 isa 指针指向一个中间类而不是真实类型。

id类型是运行时的动态类型,编译器无法知道它的真实类型,即使你发送一个id类型没有的方法,也不会产生编译警告。在实际开发过程中,我们常常指定实现某个代理的对象为id类型:

id<xxxxxxDelegate>,这是因为我们并不知道由谁实现这个代理,不知道其类型,所以就用id类型。

Class

上面的内容已经提到过Class了,先看一下它的数据结构:

typedef struct objc_class *Class;

Class其实是指向objc_class结构体的指针。这个结构体主要有以下定义:

Class super_class   这是指向父类的指针

constchar *name  类名

struct objc_ivar_list *ivars     成员变量列表

struct objc_method_list **methodLists   方法列表

struct objc_protocol_list *protocols   实现的协议列表

要注意两点:

1、objc_class中也有一个 isa 指针,这说明 Objc 类本身也是一个对象。类对象所属的类就叫做元类(Meta Class),它 表述了类对象本身所具备的元数据。

2、我们可以通过修改类的methodLists实现在运行时为类添加方法,这也是 Category 实现的原理。但Category不是真正的类,并没有isa指针,Category只会把自己的method attach到主类,并不会影响到主类的IvarList。这就是为什么分类里面不能增加成员变量的原因。参考链接:Categry为什么不能添加属性

Method

Method 代表类中某个方法的类型

typedef struct objc_method *Method;struct objc_method {

    SEL method_name                                 OBJC2_UNAVAILABLE;

    char *method_types                                OBJC2_UNAVAILABLE;    

     IMP method_imp                                    OBJC2_UNAVAILABLE;

}

介绍了IMP和SEL,这个Method就很好理解了。

Ivar

Ivar是表示成员变量的类型。

typedef struct objc_ivar *Ivar;struct objc_ivar {

char *ivar_name                                          OBJC2_UNAVAILABLE;

char *ivar_type                                            OBJC2_UNAVAILABLE;

int ivar_offset                                                OBJC2_UNAVAILABLE;

}

其中ivar_offset是基地址偏移字节

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 转至元数据结尾创建: 董潇伟,最新修改于: 十二月 23, 2016 转至元数据起始第一章:isa和Class一....
    40c0490e5268阅读 1,789评论 0 9
  • 我们常常会听说 Objective-C 是一门动态语言,那么这个「动态」表现在哪呢?我想最主要的表现就是 Obje...
    Ethan_Struggle阅读 2,232评论 0 7
  • 本文转载自:http://yulingtianxia.com/blog/2014/11/05/objective-...
    ant_flex阅读 799评论 0 1
  • runtime 运行时语言,实现Object-C的C语言库,将OC转换成C进行编译的过渡者。 作为一门动态编程语言...
    夜雨聲煩_阅读 558评论 0 0
  • 小艺2017.6.28 夏晨的雨唰唰唰,窗外青树上的鸟儿喳喳喳,一大早就被这天赖之音搅乱了那个梦。 雨晨,一个静躺...
    张镒阅读 429评论 0 1