美文网首页
Runtime的几种基本用法

Runtime的几种基本用法

作者: UILabelkell | 来源:发表于2018-11-14 17:34 被阅读8次

本方看的 (cocoachina)上一位大老的 Demo
消息机制
在OOP术语中,消息传递是指一种在对象之间发送和接收消息的通信模式。 在Objective-C中,消息传递用于在调用类和类实例的方法,即接收者接收需要执行的消息。
<pre>
// 通过类名获取类
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"));
</pre>

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

class_replaceMethod 替换类方法的定义
method_exchangeImplementations 交换两个方法的实现(具体使用案例如下)
method_setImplementation 设置一个方法的实现
注:class_replaceMethod 试图替换一个不存在的方法时候,会调用class_addMethod为该类增加一个新方法
<pre>//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....");
    }</pre>

动态加载方法
当调用一个未实现的方法,或者说发送未知的消息给接收者时候,消息的接受者会调用resolveInstanceMethod
<pre>
// 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
</pre>

消息转发

<pre>
// 第一步,消息接收者没有找到对应的方法时候,会先调用此方法,可在此方法实现中动态添加新的方法
// 返回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));
    }
    </pre>

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

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

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

//<objc>

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>

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;
    }

相关文章

  • Runtime的几种基本用法

    本方看的 (cocoachina)上一位大老的 Demo消息机制在OOP术语中,消息传递是指一种在对象之间发送和接...

  • iOS知识梳理8:万恶的Runtime

    本文中所使用的参考链接:ios开发-Runtime详解ios Runtime几种基本用法简记iOS运行时详解ios...

  • iOS Runtime 几种基本用法简记

    Runtime 介绍 这不是一遍介绍关于Runtime实现细节的文章,而是怎么利用Objective-C提供的Ru...

  • runtime基本用法

    1 当我们动态加载某个类的时候,可以用到runtime,可以便捷的获取到动态加载类的所有信息 (属性,属性名)

  • runtime基本用法

    先导入 runtime 头文件。1、用 runtime 改变变量值; 2、使用 runtime 交换方法; 3、使...

  • Runtime基本用法

    Runtime的一些术语 SEL方法选择器 它是selector在 Objc 中的表示(Swift 中是 Sele...

  • iOS中Runtime的几种基本用法记录(必看)

    Runtime顾名思义运行时,就是系统在运行的时候的一些机制,最主要的是消息机制。下面这篇文章主要给大家介绍了关于...

  • Runtime快速上手(1)

    依照惯例跳过Runtime的介绍,这里只介绍Runtime的基本用法和简单应用例子。 一、获取属性列表 输出结果:...

  • RobotFramework学习06-变量

    RobotFramework学习笔记目录 本文包含内容 变量的基本用法 共享变量 变量的用法 使用变量的几种情况 ...

  • iOS-runtime的基本用法

    iOS runtime基本用法 本内容为作者原创, 未经允许, 不得用于商业用途 我的blog 一. 改变实例变量...

网友评论

      本文标题:Runtime的几种基本用法

      本文链接:https://www.haomeiwen.com/subject/pbwjfqtx.html