runtime初体验二(实例)

废话不多说,分享两个简单的runtime使用场景:
1、在存储用户信息的时候使用的归档和解档;
2、替换系统方法。
归档和解档首先我们明确的知道需要实现两个协议方法:
解档:

- (instancetype)initWithCoder:(NSCoder *)decoder

归档:

- (void)encodeWithCoder:(NSCoder *)encoder

一般存储用户信息我们都是在model里面进行操作(因为这里是用来写用户属性的),通过KVC来进行属性的存取。
不使用runtime进行归档:

 [encoder encodeObject:self.userName forKey:@"userName"];

如果有太多太多信息要存储这里我们就需要复制粘贴很多,所以这里使用runtime会方便很多(把类拿出去就可以直接使用)。
使用runtime进行归档( 导入头文件是必须的):

class_copyIvarList(<#__unsafe_unretained Class cls#>, <#unsigned int *outCount#>)

这里的IvarList可以理解为属性列表,后面的前一个参数为哪个类,第二个参数为属性个数无符号整形指针(这里我们要传一个地址)例如:

unsigned int count = 0;
Ivar *ivars = class_copyIvarList([AccountModel class], &count);

这里的返回值可以理解为一个指针数组(只是可以这么理解,其实不是,它就是一个Ivar类型)
这里我们获取到了属性个数,然后通过下标获取属性名,紧接着通过kvc进行归档

    for (int i = 0; i < count; i++) {
        // 根据下标获取得到Ivar
        Ivar ivar = ivars[i];
        // 通过ivar_getName获取属性名称(类型为char类型 c中的类型)
        const char *name = ivar_getName(ivar);
        // 将char类型通过utf8转换成oc中的string类型
        NSString *key = [NSString stringWithUTF8String:name];
        // 然后进行归档
        [encoder encodeObject:[self valueForKey:key] forKey:key];
    }

在arc机制下释放不了c 中copy的指针,所以这里我们需要手动释放一下

free(ivars);

解档:
解档其实和上面代码基本上就一样了

        unsigned int count = 0;
        Ivar *ivars = class_copyIvarList([AccountModel class], &count);
            for (int i = 0; i < count; i++) {
            Ivar ivar = ivars[i];
            const char *name = ivar_getName(ivar);
            NSString *key = [NSString stringWithUTF8String:name];
            // 反归档
            // 这里根据属性名称作为key值来获取value值
            id value = [decoder decodeObjectForKey:key];
            [self setValue:value forKey:key];
        }

到这里归档解档就完成了,(代码直接复制到项目当中就可以使用的,不需要做任何修改)。

方法替换实例:
通常我们在发送请求的时候是这样的:

    NSURL *url = [NSURL URLWithString:@"xxxxxxxxxx"];
    NSURLRequest *request = [NSURLRequest requestWithURL:url];

如果连接中带有中文的时候这个请求就会失败:

NSURL *url = [NSURL URLWithString:@"xxxxxxxxxx/你好"];
4A3E6EDC-A18A-4F65-A09B-917B93F7E834.png

当URL中存在汉字的时候,给个提示,但是现在我不想在这里做判断,因为假如说我程序中存在好多请求,我是不是每次遇到请求都要写着个判断来进行提示?所以我们来修改系统的URLWithString方法,给NSURL类做扩展,这里我们会发现,假如我们要重写


7E960652-07D9-40EC-8C66-6763FA69B4AF.png

这个方法的话,他是个构造类方法,还是要用到URLWithString,所以这里不能重写它。
那么我们自己来构造一个方法


2EAB8712-AE95-45BD-85EE-A04559F06D36.png

接下来我们要用runtime将我们自己的方法替换系统的方法,在替换之前,我们得知道我们在哪里替换,肯定是当程序一运行我们就编译(比viewdidload先)就是load方法
    // 首先介绍下这两个方法:
    // 这个是获取对象方法
    class_getInstanceMethod(<#__unsafe_unretained Class cls#>, <#SEL name#>)
    // 这个是获取类方法
    class_getClassMethod(<#__unsafe_unretained Class cls#>, <#SEL name#>)

当然这里我们用到是肯定是类方法:
先获取系统的方法(返回类型是Method):

Method urlString = class_getClassMethod([NSURL class], @selector(URLWithString:));

在获取自己构造的方法(这里的类是NSURL,因为自己写的方法是对NSURL类做的扩展)

Method ylurlString = class_getClassMethod([NSURL class], @selector(YL_URLWithString:));

替换方法:

method_exchangeImplementations(urlString, ylurlString);

这个时候就完成了,controller中的代码一点都没有改变,还是用的URLWithString:
但是这时候运行,程序会crash,因为我们会发现,在自己写的构造方法中用到了URLWithString,当controller中运行URLWithString的时候,就会找到我们所构造的YLURLWithString,但是当走YLURLWithString里面代码的时候,又会走URLWithString,一直这样下去,造成了死循环,所以我们自己构造的方法当中应该将URLWithString改成YL_URLWithString


9B4EA95D-7429-4897-81CC-9E3DDFCF1CCC.png

这样就不会有问题了。

简单的两个应用场景,哪里说的不详情或说错的地方请谅解!runtime其实挺有意思的,但是感觉挺难得(毕竟是底层的东西),以后会更深入的去学习一下。
最后推荐一本学习runtime的书:Objective-C 2.0运行时系统编程指南.pdf

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
【社区内容提示】社区部分内容疑似由AI辅助生成,浏览时请结合常识与多方信息审慎甄别。
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

相关阅读更多精彩内容

友情链接更多精彩内容