iOS中Runtime的几种基本用法记录

Runtime 介绍

这不是一遍介绍关于Runtime实现细节的文章,而是怎么利用Objective-C提供的Runtime API进行开发的文章!

Objective-C拥有相当多的动态特性,这些特性在运行程序时候发挥作用.

Objctive-C Runtime是个运行时的库,由C和汇编实现。通过Runtime封装的C结构体和函数可以在程序运行时创建、检查和修改类以及对象及其方法,甚至可以替换或交换方法的实现。

下面记录一下关于Runtime的一些基本用法

1)消息机制

在OOP术语中,消息传递是指一种在对象之间发送和接收消息的通信模式。

在Objective-C中,消息传递用于在调用类和类实例的方法,即接收者接收需要执行的消息。

使用案例

// 通过类名获取类
Class catClass = objc_getClass("Cat"); 

//注意Class实际上也是对象,所以同样能够接受消息,向Class发送alloc消息
Cat *cat = objc_msgSend(catClass, @selector(alloc)); 

//发送init消息给Cat实例cat
cat = objc_msgSend(cat, @selector(init)); 

//发送eat消息给cat,即调用eat方法
objc_msgSend(cat, @selector(eat));

//汇总消息传递过程
objc_msgSend(objc_msgSend(objc_msgSend(objc_getClass("Cat"), sel_registerName("alloc")), sel_registerName("init")), sel_registerName("eat"));

2)方法交换 Method Swizzling

Objective-C 提供了一下API用于动态替换类方法或者实例方法的实现:

  • class_replaceMethod 替换类方法的定义
  • method_exchangeImplementations 交换两个方法的实现(具体使用案例如下)
  • method_setImplementation 设置一个方法的实现

注:class_replaceMethod 试图替换一个不存在的方法时候,会调用class_addMethod为该类增加一个新方法

使用案例

//Cat.m

+ (void)load{
 Method eatMethod = class_getInstanceMethod(self, @selector(eat));
  Method shirtMethod = class_getInstanceMethod(self, @selector(shirt));

 method_exchangeImplementations(eatMethod, shirtMethod);
}

- (void)eat{
 NSLog(@"cat eat....");
}

- (void)shirt{
 NSLog(@"cat shirt....");
}

3)动态加载方法

当调用一个未实现的方法,或者说发送未知的消息给接收者时候,消息的接受者会调用resolveInstanceMethod

使用案例

// Cat.m

//An Objective-C method is simply a C function that take at least two arguments—self and _cmd.
void run(id self, SEL _cmd, NSNumber *number){
 NSLog(@"run for %@", number);
}

//收到run:消息时候,为该类添加一个方法实现
+ (BOOL)resolveInstanceMethod:(SEL)sel{
 if(sel == NSSelectorFromString(@"run:")){
  class_addMethod(self, @selector(run:), run, "v@:@");
  return YES;
 }
 return [super resolveInstanceMethod:sel];
}

//另外针对类方法的为 resolveClassMethod

4)消息转发

// 第一步,消息接收者没有找到对应的方法时候,会先调用此方法,可在此方法实现中动态添加新的方法
// 返回YES表示相应selector的实现已经被找到,或者添加新方法到了类中,否则返回NO
+ (BOOL)resolveInstanceMethod:(SEL)sel {
 return YES;
}

// 第二步, 如果第一步的返回NO或者直接返回了YES而没有添加方法,该方法被调用
// 在这个方法中,我们可以指定一个可以返回一个可以响应该方法的对象, 注意如果返回self就会死循环
- (id)forwardingTargetForSelector:(SEL)aSelector {
 return nil;
}

// 第三步, 如果forwardingTargetForSelector:返回了nil,则该方法会被调用,系统会询问我们要一个合法的『类型编码(Type Encoding)』
// 若返回 nil,则不会进入下一步,而是无法处理消息
- (NSMethodSignature *)methodSignatureForSelector:(SEL)aSelector {
 return [NSMethodSignature signatureWithObjCTypes:"v@:"];
}

// 当实现了此方法后,-doesNotRecognizeSelector: 将不会被调用
// 在这里进行消息转发
- (void)forwardInvocation:(NSInvocation *)anInvocation {
 // 在这里可以改变方法选择器
 [anInvocation setSelector:@selector(unknown)];
 // 改变方法选择器后,需要指定消息的接收者
 [anInvocation invokeWithTarget:self];
}

- (void)unknown {
 NSLog(@"unkown method.......");
}

// 如果没有实现消息转发 forwardInvocation 则调用此方法
- (void)doesNotRecognizeSelector:(SEL)aSelector {
 NSLog(@"unresolved method :%@", NSStringFromSelector(aSelector));
}

注: 『类型编码(Type Encoding)』

5)动态关联属性

对象在内存中的排布可以看成一个结构体,该结构体的大小并不能动态变化,所以无法在运行时动态给对象增加成员变量。相对的,对象的方法定义都保存在类的可变区域中。

如下图所示为Class 的描述信息,其中methodList为可访问类中定义的方法的指针的指针,通过修改该指针所指向的指针的值,我们可以实现为类动态增加方法实现。

因此,我们可以实现动态为一个类增加成员方法,但是却不能直接为类增加成员变量,这就是category的实现原理。

//<objc/runtime.h>

struct objc_class {
 Class isa OBJC_ISA_AVAILABILITY;

#if !__OBJC2__
 Class super_class          OBJC2_UNAVAILABLE;
 const char *name           OBJC2_UNAVAILABLE;
 long version            OBJC2_UNAVAILABLE;
 long info            OBJC2_UNAVAILABLE;
 long instance_size          OBJC2_UNAVAILABLE;
 struct objc_ivar_list *ivars        OBJC2_UNAVAILABLE;
 struct objc_method_list **methodLists     OBJC2_UNAVAILABLE;
 struct objc_cache *cache         OBJC2_UNAVAILABLE;
 struct objc_protocol_list *protocols      OBJC2_UNAVAILABLE;
#endif

} OBJC2_UNAVAILABLE;

使用案例

//Cat+Extend.h

@interface Cat (extend)

@property(nonatomic, copy) NSString *name;

@end

//Cat+Extend.m

@implementation Cat (extend)

- (void)setName:(NSString *)name{
 objc_setAssociatedObject(self, "name", name, OBJC_ASSOCIATION_RETAIN_NONATOMIC);
}

- (NSString *)name{
 return objc_getAssociatedObject(self, "name");
}

@end

6)字典转模型应用

通过Class的结构体内容,可以看到ivars指针指向包含了类中成员变量的结构体,通过它可以得到类中定义的成员变量,而Objective-C中提供了相应的API方法: class_copyIvarList

//<objc/runtime.h>

struct objc_class {
 Class isa OBJC_ISA_AVAILABILITY;

#if !__OBJC2__
 Class super_class          OBJC2_UNAVAILABLE;
 const char *name           OBJC2_UNAVAILABLE;
 long version            OBJC2_UNAVAILABLE;
 long info            OBJC2_UNAVAILABLE;
 long instance_size          OBJC2_UNAVAILABLE;
 struct objc_ivar_list *ivars        OBJC2_UNAVAILABLE;
 struct objc_method_list **methodLists     OBJC2_UNAVAILABLE;
 struct objc_cache *cache         OBJC2_UNAVAILABLE;
 struct objc_protocol_list *protocols      OBJC2_UNAVAILABLE;
#endif

} OBJC2_UNAVAILABLE;

使用案例

//Cat.h

@property(nonatomic, copy) NSString *cid;

@property(nonatomic, copy) NSString *age;

+ (instancetype)modelWithDict:(NSDictionary *)dict;

//Cat.m

+ (instancetype)modelWithDict:(NSDictionary *)dict{
 id model = [[self alloc] init];
 unsigned int count = 0;

 Ivar *ivars = class_copyIvarList(self, &count);
 for (int i = 0 ; i < count; i++) {
  Ivar ivar = ivars[i];

  NSString *ivarName = [NSString stringWithUTF8String:ivar_getName(ivar)];

  //这里注意,拿到的成员变量名为_cid,_age
  ivarName = [ivarName substringFromIndex:1];
  id value = dict[ivarName];

  [model setValue:value forKeyPath:ivarName];
 }

 return model;
}

总结

以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,如果有疑问大家可以留言交流,谢谢大家对我们的支持。

(0)

相关推荐

  • iOS runtime动态添加方法示例详解

    前言 上手开发 iOS 一段时间后,我发现并不能只着眼于完成需求,利用闲暇之余多研究其他的开发技巧,才能在有限时间内提升自己水平.当然,"其他开发技巧"这个命题对于任何一个开发领域都感觉不找边际,而对于我来说,尝试接触 objc/runtime 不失为是开始深入探索 iOS 开发的第一步.下面主要介绍了关于iOS runtime动态添加方法的相关内容,分享出来供大家参考学习,下面话不多说了,来一起看看详细的介绍吧. 一.概念 1.动态添加方法 开发使用场景:如果一个类方法非常多,加载类

  • 总结iOS中runtime的使用

    做iOS的朋友都知道或听说runtime,这个东西很像java的反射机制,但功能远胜于java的反射.通过runtime我们可以动态的向一个类中添加属性.成员变量.方法,以及对其进行读写访问. 一.runtime简介 RunTime简称运行时.OC就是运行时机制,也就是在运行时候 的一些机制,其中最主要的是消息机制. 对于C语言,函数的调用在编译的时候会决定调用哪个函数. 对于OC的函数,属于动态调用过程,在编译的时候并不能决定真正调用哪个函数,只有在真正运行的时候才会根据函数的名称 找到对应的

  • iOS利用Runtime实现友盟页面数据统计的功能示例

    前言 一般项目中集成统计功能随因产品类型不同而使用功能不同,但大多数统计一般只有一个目的,就是记录用户习惯,研究用户习惯,从而为用户带来更好的体验,本文主要介绍了关于iOS用Runtime实现友盟页面数据统计功能的相关内容,分享出来供大家参考学习,下面话不多说了,来一起看看详细的介绍吧. 一.概念 1.实现页面的统计,需要在每一个类中实现这个方法: - (void)viewWillAppear:(BOOL)animated { [super viewWillAppear:animated]; [

  • iOS通过Runtime实现友盟统计的实例代码

    在友盟官网可以看到相应的步骤,申请appkey,导入SDK,然后在AppDelegate里面写入相应的代码,下面就是关键的代码: 实现页面的统计需要在每个UIViewController中配对调用如下方法: - (void)viewWillAppear:(BOOL)animated { [super viewWillAppear:animated]; [MobClick beginLogPageView:@"PageOne"];//("PageOne"为页面名称,可

  • iOS runtime forwardInvocation详解及整理

     iOS runtime forwardInvocation详解 代码: TestModel - (NSMethodSignature *)methodSignatureForSelector:(SEL)aSelector { if(aSelector == @selector(testMethod)) { return [NSMethodSignature signatureWithObjCTypes:"v@:"]; } return nil; } -(void)forwardInv

  • IOS 中runtime使用方法整理

    IOS 中runtime使用方法整理 做iOS的朋友都知道或听说runtime,这个东西很像java的反射机制,但功能远胜于java的反射.通过runtime我们可以动态的向一个类中添加属性.成员变量.方法,以及对其进行读写访问. 新建两个类ClassOne和ClassTwo #import <Foundation/Foundation.h> @interface ClassOne : NSObject{ NSString *_publicVar1; NSString *_publicVar2

  • iOS Runtime详解(新手也看得懂)

    前言 Runtime的特性主要是消息(方法)传递,如果消息(方法)在对象中找不到,就进行转发,具体怎么实现的呢.我们从下面几个方面探寻Runtime的实现机制. Runtime介绍 Runtime消息传递 Runtime消息转发 Runtime应用 Runtime介绍 Objective-C 扩展了 C 语言,并加入了面向对象特性和 Smalltalk 式的消息传递机制.而这个扩展的核心是一个用 C 和 编译语言 写的 Runtime 库.它是 Objective-C 面向对象和动态机制的基石.

  • iOS使用runtime修改文本框(TextField)的占位文字颜色

    前言 在大家平时的开发中,有时候会遇到背景深色的界面上需要添加TextField,TextField默认的占位颜色也是深灰色,比较难看清,这时候就需要修改展位文字的颜色,可是系统没有提供相应的方法,那么就需要我们自己来自定义了 修改后的效果 话不多说,下面上代码: // .h文件 #import <UIKit/UIKit.h> @interface UITextField (Placeholder) @property UIColor *placeholderColor; @end // .m

  • iOS runtime知识梳理

    一.runtime简介 RunTime简称运行时.OC就是运行时机制,也就是在运行时候的一些机制,其中最主要的是消息机制. 对于C语言,函数的调用在编译的时候会决定调用哪个函数. 对于OC的函数,属于动态调用过程,在编译的时候并不能决定真正调用哪个函数,只有在真正运行的时候才会根据函数的名称找到对应的函数来调用. 事实证明: 在编译阶段,OC可以调用任何函数,即使这个函数并未实现,只要声明过就不会报错. 在编译阶段,C语言调用未实现的函数就会报错. 二.runtime作用 1.发送消息 ▪ 方法

  • IOS Object-C 中Runtime详解及实例代码

    IOS Object-C 中Runtime详解 最近了解了一下OC的Runtime,真的是OC中很强大的一个机制,看起来比较底层,但其实可以有很多活用的方式. 什么是Runtime 我们虽然是用Objective-C写的代码,其实在运行过程中都会被转化成C代码去执行.比如说OC的方法调用都会转成C函数 id objc_msgSend ( id self, SEL op, - ); 而OC中的对象其实在Runtime中都会用结构体来表示,这个结构体中包含了类名.成员变量列表.方法列表.协议列表.缓

随机推荐