iOS中的runtime添加方法

runtime是面试当中经常会问到的一个问题,可是这个在平常当中使用确实不多,至少本人是这样,以前一直只知道有那么个东西,一个没尝试过,现在来看看。

 

runtime的优点:大家都知道OC是一个动态语言,那runtime就是实现动态的一个方式,可以动态生成类,添加方法和属性。动态生成的好处:

  1. 可以优化内存,类似于懒加载那样,在需要的时候加载到内存
  2. 交换方法实现(交换系统的方法)
  3. 遍历类的所有成员变量(修改textfield的占位文字颜色、字典转模型、自动归档解档)
  4. 利用关联对象(AssociatedObject)给分类添加属性
  5. 可用统计和处理传错方法或未实现的方法

 

动态添加方法:

如果是正常的方法调用,OC中应该是生成一个对象a,调用test方法:[a test],但是如果对象a没有test方法,这样写编译就会报错。现在我们新建一个Dynamic类,继承NSObject,现在我们需要Dynamic的一个对象运行test方法,但是我们不需要任何的声明,在调用的时候,我们需要用到performSelector方法:

Dynamic *dynamic = [[Dynamic alloc] init];

 [dynamic performSelector:@selector(test)];

这样写的话会有一个警告,说明你的类中并没有test方法,在Dynamic.h的方法中不需要添加任何东西,Dynamic.m中如下:

#import "Dynamic.h"

#import <objc/runtime.h>

 

@implementation Dynamic

 

+ (BOOL)resolveInstanceMethod:(SEL)sel {

    if (sel = @selector(test)) {

        class_addMethod([self class], sel, class_getMethodImplementation(self, @selector(implement)), "v@:");

        return YES;

    }

    return [super resolveInstanceMethod:sel];

}

 

- (void)implement {

    NSLog(@"动态添加了一个方法");

}

 

这样就可以动态完成了对于Dynamic类的test方法的动态添加,下面我会详细讲一讲里面的内容。

当我们用到 [dynamic performSelector:@selector(test)]时,系统由于找不到test方法,会调用Dynamic类的

+ (BOOL)resolveInstanceMethod:(SEL)sel ;

这个方法是当类对象调用方法时用到的,还有一个比较相像的,+ (BOOL)resolveClassMethod:(SEL)sel;这个是调用类方法时会进入的,可以动态添加类方法。

这里注意下调用顺序,当方法有实现时,是先进入方法的实现里,然后进入resolveInstanceMethod方法,对象执行任何方法最后都会进入resolveInstanceMethod方法。

 

下面就是class_addMethod(Class  _Nullable __unsafe_unretained cls, SEL  _Nonnull name, IMP  _Nonnull imp, const char * _Nullable types),这个就是动态添加的runtime中方法,下面讲解4个参数:

* @param cls The class you want to modify.

* @param name A selector that identifies the method whose implementation you want to replace.

* @param imp The new implementation for the method identified by name for the class identified by cls.

* @param types An array of characters that describe the types of the arguments to the method.

这是苹果对于参数的解释

1、Class  _Nullable __unsafe_unretained cls        类对象,给哪个类添加方法

2、SEL  _Nonnull name       方法名,哪个方法名是你想去替换实现的

3、IMP  _Nonnull imp      替换的实现,注意这里IMP是方法的指针

4、const char * _Nullable types      参数类型,这里的有一种特殊的表达方式,无返回值不带参数的为"v@:",v代表void,@代表对象,:代表方法,但是你看我上面的代码,并没有一个参数,这是因为每一个方法会默认隐藏两个参数,self、_cmd,self代表方法调用者,_cmd代表这个方法的SEL,其他的缩写看下面的链接。

苹果文档

然后就是写下面的方法实现了,就完成了。对于方法的实现,苹果文档中是用C语言写的,不知道为什么,可能用C有更好的效率,也希望有知道的人留言。C语言实现方法的,第三个参数应该 (IMP) implement 这样书写。

 

上面的实现是无参数的,当你需要传入参数时,有performSelector: withObject:这个方法,最多可以传2个参数,传三个以上参数也有方法,自行百度。

多传参数后,方法的实现也需要输入对应的参数,class_addMethod的第四个参数也需要添加相应个数的@,不然的话会崩溃。动态添加方法基本就是这些了。

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
iOS Runtime是一种运行时环境,它是iOS操作系统的重要组成部分。iOS Runtime为开发者提供了一套动态询问、修改和扩展应用程序行为的机制。 首先,iOS Runtime实现了Objective-C语言的动态特性。Objective-C是一种面向对象的编程语言,它具有动态特性,即在运行时能够动态地修改对象的行为。iOS Runtime允许开发者通过运行时系统对类、对象、方法以及属性进行动态操作。例如,开发者可以在运行时为某个类添加新的方法,也可以通过运行时修改类的实例变量。 其次,iOS Runtime还提供了消息发送机制。在Objective-C,对象之间的通信是通过消息发送来完成的。iOS Runtime负责将消息转发给正确的接收者,并执行相应的方法。这个过程是动态的,开发者可以在运行时动态改变消息的接收者或方法的实现。这为Objective-C语言带来了很高的灵活性。 此外,iOS Runtime还支持方法的交换与替换。开发者可以通过运行时机制,在运行时将一个方法的实现替换为另一个方法的实现,或者交换两个方法的实现。这对于调试、性能监控和代码扩展都是非常有用的。 总结来说,iOS RuntimeiOS操作系统实现Objective-C语言特性的核心组件。它提供了动态特性的支持,使开发者能够在运行时动态地修改类、对象和方法的行为,实现更加灵活和可扩展的应用程序。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值