Runtime基础运用实践和总结

runtime的理解

今天同学问我什么是runtime,我想了想就简单总结出了这篇文章,以下仅供参考.

我所理解的runtime是一个使用C编写的库,为C添加了面向对象的特性,它是一个库(Runtime Library中文:运行时库).在这个库中可以用C函数来实现方法,对象也可以用C语言的结构体来表示...所有oc的方法的背后都是通过runtime来运行的.

如图创建一个Person

Snip20170608_1.png

  通过终端编译一下.m :
  1.终端切换到工程的目录下
  2.然后输入命令clang -rewrite-objc main.m
  3.找到编译后的main.cpp如下图

Snip20170608_2.png

  然后我点开cpp文件(C++文件)发现一大推代码,在最后发现了如下图的代码:


Snip20170608_3.png

结论:

Person *p = [[Person alloc] init]最终被编译成了下面的样子(也就是上图中的样子)

Person *p = ((Person *(*)(id, SEL))(void *)objc_msgSend)((id)((Person *(*)(id, SEL))(void *)objc_msgSend)((id)objc_getClass("Person"), sel_registerName("alloc")), sel_registerName("init"));

我发现调用方法其实最终被编译成发送具体某一个消息.

runtime如何应用,怎么用?

先说用处:
1.拦截系统自带的方法调用(好多人用这种方法省去了很多重复性的操作,比如拦截imageName方法,不同系统版本环境下对图片做不同操作)
2.给分类增加个属性(我的渐变色背景二那篇文章里面的代码有用到一些)
3.归档解档

一、拦截系统自带的方法,调用自己的方法

导入<objc/runtime.h>
需要用到

//获取类方法
Method class_getClassMethod(Class cls , SEL name)
//获取对象方法
Method class_getInstanceMethod(Class cls , SEL name)
//交换方法
void method_exchangeImplementations(Method m1 , Method m2)

具体场景
需求:iOS7 后需要版本适配,不同系统版本使用不同样式图片(拟物化和扁平化),如何通过不去手动一个个修改每个UIImage的imageNamed:方法就可以实现为该方法中加入版本判断语句?

步骤:
1.创建UIImage分类
2.m里自定义一个方法
3.在load里做交换方法(load只执行一次)
4.如下图;

Snip20170608_4.png

实现之后,我在VC里调用下imageNamed方法如下:

Snip20170608_5.png

通过上面的实现就拦截到了系统方法针对不同系统版本做我想做的操作.

二、给分类设置属性

代码可以参考渐变色背景(二)那篇文章,这里我对方法总结一下

//set方法里实现
//id object 哪一个对象设置属性
//const void *key 通过key取出存的value值
//value 给属性设置的值(存的value值)
//policy: 
//policy设置的值和assgin copy retain差不多,只不过变成:
//`OBJC_ASSOCIATION_ASSIGN`,
//`OBJC_ASSOCIATION_COPY_NONATOMIC`,
//`OBJC_ASSOCIATION_RETAIN_NONATOMIC`

void objc_setAssociatedObject(id object , const void *key ,id value ,objc_AssociationPolicy policy)

//get方法里实现,对应的参数说明看set
id objc_getAssociatedObject(id object , const void *key)
三、NSCode协议下,归档解档
//获取所有成员变量和变量总数
Ivar *class_copyIvarList(Class cls , unsigned int *outCount)
//获得成员变量的名字
const char *ivar_getName(Ivar v)
//获得成员变量的类型
const char *ivar_getTypeEndcoding(Ivar v)

//归档
- (void)encodeWithCoder:(NSCoder *)coder
{
    //属性的个数
    unsigned int count = 0;
    //
    Ivar * ivars = class_copyIvarList([self class], &count);
    for (int i = 0; i < count; i++) {
       //取出对应的Ivar
        Ivar ivar = ivars[i];
        //拿到名称
        const char * name = ivar_getName(ivar);
        //OC 字符串
        NSString * key = [NSString stringWithUTF8String:name];
        //归档
        [coder encodeObject:[self valueForKey:key] forKey:key];
    }
    free(ivars);
}

//解档
- (instancetype)initWithCoder:(NSCoder *)coder
{
    if (self = [super init]) {
        //属性的个数
        unsigned int count = 0;
        Ivar * ivars = class_copyIvarList([self class], &count);
        for (int i = 0; i < count; i++) {
            //取出对应的Ivar
            Ivar ivar = ivars[i];
            //拿到名称
            const char * name = ivar_getName(ivar);
            //OC 字符串
            NSString * key = [NSString stringWithUTF8String:name];
            //解档
            id value = [coder decodeObjectForKey:key];
            //KVC--设置值到属性上面!!!
            [self setValue:value forKey:key];
        }
        free(ivars);
    }
    return self;
}

我日后再深入了解,及时更新文章,thank you

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 对于从事 iOS 开发人员来说,所有的人都会答出【runtime 是运行时】什么情况下用runtime?大部分人能...
    梦夜繁星阅读 3,734评论 7 64
  • 参考链接: http://www.cnblogs.com/ioshe/p/5489086.html 简介 Runt...
    乐乐的简书阅读 2,166评论 0 9
  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 135,010评论 19 139
  • W姑娘:所爱隔山海,山海不可平。 W姑娘这里的天气干燥寒冷,却又晴朗依旧。此刻的她有了珍贵的闲暇时光给自己...
    无恙S阅读 331评论 0 2
  • 动物园班级 看到这个题目,你一定很好奇,为什么把班级比做了动物园呢?我自有这样说的道理,下面,慢慢给你解释。 大...
    转角遇见美阅读 707评论 0 1