iOS RunTime 动态添加对象方法和类方法

2020-12-25  本文已影响0人  陶小亮

小编的语言表达能力不是很强,希望看到的大佬能理解我的意思,如果有什么问题欢迎各位大佬评论区留言。

消息转发机制相信大家都有所了解,在此不做过多的赘述,如有想了解的,请查看下图

感谢小马哥

然后我这里讲下如果这里的消息发送没有任何执行的话,Runtime会进行一步动态方法解析。

怎么解析???

首先我们创建一个继承自NSObject的Person类(tip:我也不知道为什么都喜欢叫Person类)

person.h

.m文件

person.m

我这里并没有实现test方法。

接着我在HomeController里面创建Person对象并调用person的test方法。

来,我们运行下,会发现一个很经典的错误:

然后我们利用runTime去给动态添加对象方法:我们进入到Person类的.m文件里面,导入runtime头文件。添加一个返回Bool值的方法:+ (BOOL)resolveInstanceMethod:(SEL)sel。这个方法 是解决对象方法,参数sel是要解决的方法名称。这个方法是一旦发现对象方法在消息发送的过程中没有找到之后调用的。我们在这里处理这里的sel.

1.首先我们要判断下当前的sel是不是我们的test方法,因为有可能有其他的方法也是没有实现。

2.然后如果这个sel是test方法的话,我们需要对这个方法进行处理,其实这里我理解的是对这个方法进行一次转换,实质就是,一旦我发现这个sel没有实现,那我就去实现另一个方法。

3.因为我们都知道,对象方法是添加到类信息里面的。

这里发现,我们需要传入一个新的方法。

///调用了新的方法

- (void)otherTest{

    NSLog(@"%s",__func__);

}

总结 1、2、3点之后,我们得出以下方法

///动态添加方法

+ (BOOL)resolveInstanceMethod:(SEL)sel{

    if(sel ==@selector(test)) {

        ///如果要处理的方法是test

        ///添加对象方法

        class_addMethod(<#Class  _Nullable __unsafe_unretained cls#>, <#SEL  _Nonnull name#>, <#IMP  _Nonnull imp#>, <#const char * _Nullable types#>)

    }

    return YES;

}

///这里的class_addMethod方法的第一个参数是当前类,传self

///这里的class_addMethod方法的第二个参数是需要解决的方法名称,传sel

///这里的class_addMethod方法的第三个参数是需要转换成新的方法实现,IMP类型,目前不知道传啥

///这里的class_addMethod方法的第四个参数是转换成新的方法type,目前不知道传啥

class_addMethod(<#Class  _Nullable __unsafe_unretained cls#>, <#SEL  _Nonnull name#>, <#IMP  _Nonnull imp#>, <#const char * _Nullable types#>)

我们要传class_addMethod的第三和第四个参数。其实就是把otherTest信息传过去。

我们利用runTime获取到otherTest方法的Method对象:

Method method = class_getInstanceMethod(self, @selector(otherTest));

获取到otherTest的IMP和Types

method_getImplementation(method),

method_getTypeEncoding(method)

于是得到+ (BOOL)resolveInstanceMethod:(SEL)sel实现如下

+ (BOOL)resolveInstanceMethod:(SEL)sel{

    if(sel ==@selector(test)) {

        ///如果要处理的方法是test

        Methodmethod =class_getInstanceMethod(self,@selector(otherTest));

        ///添加对象方法

        class_addMethod(self, sel, method_getImplementation(method), method_getTypeEncoding(method));

        ///根据李明杰老师的教程阅读源码之后发现,这里返回YES或者NO都不影响

        returnYES;

    }

    //实现父类的方法

    return [super resolveInstanceMethod:sel];

}

最终处理结果如下图所示:

到这里我们就动态的给ZLPerson添加了test方法。我们Run一下

OK没问题。

其实这里涉及到很多的RunTime的底层知识,作者这里由于对RunTime底层的数据结构了解的不是很深,就没有深入的解析。如果有阅读不是很舒服的地方,或者有错误的地方,请给位大佬评论区指教,感谢。

这里附上给类对象动态添加方法的图片(原理和对实例对象动态添加方法一致)

ZLPerson.m ZLPerson.h 调用
上一篇下一篇

猜你喜欢

热点阅读