run time简单理解

基本消息传递

在面向对象编程中,对象调用方法叫做发送消息。在编译时,程序的源代码就会从对象发送消息转换成Runtime的objc_msgSend函数调用。
例如某实例变量receiver实现某一个方法oneMethod

[receiver oneMethod];

Runtime会将其转成类似这样的代码

objc_msgSend(receiver, selector);

具体会转换成什么代码呢?
Runtime会根据类型自动转换成下列某一个函数:
objc_msgSend:普通的消息都会通过该函数发送
objc_msgSend_stret:消息中有数据结构作为返回值(不是简单值)时,通过此函数发送和接收返回值
objc_msgSendSuper:和objc_msgSend类似,这里把消息发送给父类的实例
objc_msgSendSuper_stret:和objc_msgSend_stret类似,这里把消息发送给父类的实例并接收返回值



objc_msgSend函数的调用过程:

  • 第一步:检测这个selector是不是要忽略的。
  • 第二步:检测这个target是不是nil对象。nil对象发送任何一个消息都会被忽略掉。
  • 第三步:
    1.调用实例方法时,它会首先在自身isa指针指向的类(class)methodLists中查找该方法,如果找不到则会通过class的super_class指针找到父类的类对象结构体,然后从methodLists中查找该方法,如果仍然找不到,则继续通过super_class向上一级父类结构体中查找,直至根class;
    2.当我们调用某个某个类方法时,它会首先通过自己的isa指针找到metaclass,并从其中methodLists中查找该类方法,如果找不到则会通过metaclass的super_class指针找到父类的metaclass对象结构体,然后从methodLists中查找该方法,如果仍然找不到,则继续通过super_class向上一级父类结构体中查找,直至根metaclass;
  • 第四部:前三部都找不到就会进入动态方法解析(看下文)。

消息动态解析

  • 第一步:通过resolveInstanceMethod:方法决定是否动态添加方法。如果返回Yes则通过class_addMethod动态添加方法,消息得到处理,结束;如果返回No,则进入下一步;
  • 第二步:这步会进入forwardingTargetForSelector:方法,用于指定备选对象响应这个selector,不能指定为self。如果返回某个对象则会调用对象的方法,结束。如果返回nil,则进入第三部;
  • 第三部:这步我们要通过methodSignatureForSelector:方法签名,如果返回nil,则消息无法处理。如果返回methodSignature,则进入下一步;
  • 第四部:这步调用forwardInvocation:方法,我们可以通过anInvocation对象做很多处理,比如修改实现方法,修改响应对象等,如果方法调用成功,则结束。如果失败,则进入doesNotRecognizeSelector方法,若我们没有实现这个方法,那么就会crash。



/ 动态创建对象 创建一个Person 继承自 NSObject类

        Class People = objc_allocateClassPair([NSObject class], "Person", 0);

 

        // 为该类添加NSString *_name成员变量

        class_addIvar(People, "_name", sizeof(NSString*), log2(sizeof(NSString*)), @encode(NSString*));

        // 为该类添加int _age成员变量

        class_addIvar(People, "_age", sizeof(int), sizeof(int), @encode(int));

 

        // 注册方法名为say的方法

        SEL s = sel_registerName("say:");

        // 为该类增加名为say的方法

        class_addMethod(People, s, (IMP)sayFunction, "v@:@");

 

        // 注册该类

        objc_registerClassPair(People);

 

        // 创建一个类的实例

        id peopleInstance = [[People alloc] init];

 

        // KVC 动态改变 对象peopleInstance 中的实例变量

        [peopleInstance setValue:@"苍老师" forKey:@"name"];

 

        // 从类中获取成员变量Ivar

        Ivar ageIvar = class_getInstanceVariable(People, "_age");

        // 为peopleInstance的成员变量赋值

        object_setIvar(peopleInstance, ageIvar, @18);

 

        // 调用 peopleInstance 对象中的 s 方法选择器对于的方法

        // objc_msgSend(peopleInstance, s, @"大家好!"); // 这样写也可以,请看我博客说明

        ((void (*)(id, SEL, id))objc_msgSend)(peopleInstance, s, @"大家好");

 

        peopleInstance = nil; //当People类或者它的子类的实例还存在,则不能调用objc_disposeClassPair这个方法;因此这里要先销毁实例对象后才能销毁类;

 

        // 销毁类

        objc_disposeClassPair(People);

 

    }

    return 0;

}


最后的结果是:18岁的苍老师说:大家好!

在使用


objc_msgSend(peopleInstance, s, @"大家好!");


默认会出现以下错误:

objc_msgSend()报错Too many arguments to function call ,expected 0,have3

直接通过objc_msgSend(self, setter, value)是报错,说参数过多。

请这样解决:

Build Setting–> Apple LLVM 7.0 – Preprocessing–> Enable Strict Checking of objc_msgSend Calls 改为 NO


评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值