美文网首页
自己练习runtime

自己练习runtime

作者: 简单也好 | 来源:发表于2017-05-12 14:51 被阅读150次

将近一年没有更新文章了,但是还是有人看我的文章,点赞和关注,一直想写,却因为各种原因拖着了。
如今年后入职了新的公司,必须再写点什么了。
如今iOS的求职市场已经世风日下,很多公司招聘一条计算机专业毕业的要求,就难住了不少培训出来的同学,然而归根结底,技术扎实过硬,总有碗饭吃不是。

今天就说说面试几乎必问的runtime机制
虽然网上文章很多,比如这篇很好http://blog.csdn.net/liangliang103377/article/details/39007683
但还是自己敲一敲研究一下,自己的demo地址
https://github.com/DaLiWangCC/MyStudy

Runtime是一些常用第三方实现的基础,比如MJExtension和YYModel

先说说runtime

复制一段

runtime(简称运行时),是一套 纯C(C和汇编写的) 的API。而 OC 就是 运行时机制,也就是在运行时候的一些机制,其中最主要的是 消息机制。
对于 C 语言,函数的调用在编译的时候会决定调用哪个函数。
OC的函数调用成为消息发送,属于 动态调用过程。在编译的时候并不能决定真正调用哪个函数,只有在真正运行的时候才会根据函数的名称找到对应的函数来调用。
事实证明:在编译阶段,OC 可以 调用任何函数,即使这个函数并未实现,只要声明过就不会报错,只有当运行的时候才会报错,这是因为OC是运行时动态调用的。而 C 语言 调用未实现的函数 就会报错。
基本类型

再看看源码

//这四个属性分别可以指向一个方法、实例变量、类目、属性

1.typedef struct objc_method *Method;
// An opaque type that represents a method in a class definition.
//定义对象方法或类方法。这个类型提供了方法的名字(就是**选择器**)、参数数量和类型,以及返回值(这些信息合起来称为方法的**签名**),还有一个指向代码的函数指针(也就是方法的**实现**)。
// 结构
struct objc_method {
    SEL method_name                                          OBJC2_UNAVAILABLE;
    char *method_types                                       OBJC2_UNAVAILABLE;
    IMP method_imp                                           OBJC2_UNAVAILABLE;
}                                    
                        
2.typedef struct objc_ivar *Ivar;
/// An opaque type that represents an instance variable.
// 一个不透明的类型,定义了变量的实例对象
// 结构
struct objc_ivar {
    char *ivar_name                                          OBJC2_UNAVAILABLE;
    char *ivar_type                                          OBJC2_UNAVAILABLE;
    int ivar_offset                                          OBJC2_UNAVAILABLE;
#ifdef __LP64__
    int space                                                OBJC2_UNAVAILABLE;
#endif
}   

3.typedef struct objc_category *Category; 
/// An opaque type that represents a category. 这个不知道什么情况会用

4.typedef struct objc_property *objc_property_t;
/// An opaque type that represents an Objective-C declared property.

// 接着两个类型
SEL 
//定义选择器。选择器是方法名的唯一标识符
IMP  
//定义方法实现。这只是一个指向某个函数的指针,该函数接受一个对象、一个选择器和一个可变长参数列表(varargs),返回一个对象

常用方法

// 得到类的所有方法
    Method *allMethods = class_copyMethodList([Person class], &count);
// 得到所有成员变量
    Ivar *allVariables = class_copyIvarList([Person class], &count);
// 得到所有属性
    objc_property_t *properties = class_copyPropertyList([Person class], &count);
// 根据名字得到类变量的Ivar指针,但是这个在OC中好像毫无意义
Ivar oneCVIvar = class_getClassVariable([Person class], name);
// 根据名字得到实例变量的Ivar指针
    Ivar oneIVIvar = class_getInstanceVariable([Person class], name);
// 找到后可以直接对私有变量赋值
    object_setIvar(_per, oneIVIvar, @"Mike");//强制修改name属性
/* 动态添加方法:
     第一个参数表示Class cls 类型;
     第二个参数表示待调用的方法名称;
     第三个参数(IMP)myAddingFunction,IMP是一个函数指针,这里表示指定具体实现方法myAddingFunction;
     第四个参数表方法的参数,0代表没有参数;
     */
    class_addMethod([_per class], @selector(sayHi), (IMP)myAddingFunction, 0);
// 交换两个方法
    method_exchangeImplementations(method1, method2);

// 关联两个对象
objc_setAssociatedObject(id object, const void *key, id value, objc_AssociationPolicy policy)
/*
 id object                     :表示关联者,是一个对象,变量名理所当然也是object
 const void *key               :获取被关联者的索引key
 id value                      :被关联者,这里是一个block
 objc_AssociationPolicy policy : 关联时采用的协议,有assign,retain,copy等协议,一般使用OBJC_ASSOCIATION_RETAIN_NONATOMIC
*/

给一个category添加属性

这个问题面试很常问
如果在category中添加实例变量,会直接报错
instance variables may not be placed in categories

如果在category中添加属性
@property (nonatomic,assign) float height;
使用时会报错 [Person setHeight:]: unrecognized selector sent to instance
category没有实现对应属性的set方法
所以关键是给他写上set get方法

category不能添加属性的原因(知乎的解释)

struct objc_category {
    char *category_name                                      OBJC2_UNAVAILABLE;
    char *class_name                                         OBJC2_UNAVAILABLE;
    struct objc_method_list *instance_methods                OBJC2_UNAVAILABLE;
    struct objc_method_list *class_methods                   OBJC2_UNAVAILABLE;
    struct objc_protocol_list *protocols                     OBJC2_UNAVAILABLE;
}   

category里面虽然可以添加 property,但是这些 properties 并不会自动生成 Ivar,也就是不会有 @synthesize 的作用,dyld 加载的期间,这些 categories 会被加载并 patch 到相应的类中。这个过程是一个动态过程,Ivar 不能动态添加,因为表示 ObjC 类的结构体运行时并不能改变。

实现set和get方法,就完成了在类目中添加属性

- (void)setHeight:(float)height
{
    NSNumber *num = [NSNumber numberWithFloat:height];
    objc_setAssociatedObject(self, @"addHeight", num, OBJC_ASSOCIATION_ASSIGN);
}

//提取属性的值
- (float)height
{
    NSNumber *number = objc_getAssociatedObject(self, @"addHeight");
    return [number floatValue];
}

相关文章

  • 自己练习runtime

    将近一年没有更新文章了,但是还是有人看我的文章,点赞和关注,一直想写,却因为各种原因拖着了。如今年后入职了新的公司...

  • p4学习笔记-3

    接下来是p4Runtime的尝试 使用P4Runtime实现控制平面 介绍 在本练习中,我们将使用P4Runtim...

  • Runtime的小练习

    用block封装后的UIAlertView感觉更好用了!

  • runtime的简单练习

    很喜欢简书的整体设计模式,于是把以前的笔记稍加整理,搬一部分到这里。 通过runtime实现简单的json解析(仿...

  • 练习一下runtime关联属性

    刚自己练习了一下runtime属性关联写了一个这个通过cashapLayer 与贝塞尔曲线进行配合绘制出形状 使用...

  • 9.28 runtime hook散记

    runtime runtime是一种机制,比如一个老板做好一道菜。如果你会用runtime,就如你可以自己加自己的...

  • objective-C之runtime第一篇

    网上查看了一些runtime的资料, 关于runtime是什么?这里想用自己的理解,简单点说:runtime是一套...

  • iOS之RunTime探索与实践

    Runtime 概念 Runtime 相关概念 Runtime 实践 Runtime概念 Runtime简称运行时...

  • RunTime的使用和介绍

    自己比较懒 网上搜集了下runtime相关的总结,学习学习runtime的简介runtime是一套比较底层的纯C语...

  • 微点滴:对于OC的runtime运行时的一些理解

    runtime 的一些理解, 整理收集了不少网络的知识, 只是用于自己对runtime的理解,以备以后自己查阅...

网友评论

      本文标题:自己练习runtime

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