美文网首页runtime相关程序猿大本营runtime
iOS runtime从菜鸟到应用(小灰进阶篇)

iOS runtime从菜鸟到应用(小灰进阶篇)

作者: 睡不着的叶 | 来源:发表于2016-07-12 17:39 被阅读1865次
    3只小灰

    前言

    书接上回,经过了小白理论篇,相信大家对于runtime是什么能有一个大体的概念了,恭喜你装逼神技已经加了一点技能点了。同时也很荣幸,再次感谢简书的小编能给我拉到首页。然后这篇文章,介绍一下如何应用,加深一下理解,大家共同进步。


    there is 正文

    1.交换方法

    • 使用场景:系统自带的方法功能不够,给系统自带的方法扩展一些功能,并且保持原有的功能。(可以和继承系统类,重写方法达到一样效果)

    -(void)viewDidLoad {
    [super viewDidLoad];
    // Do any additional setup after loading the view, typically from a nib.
    // 需求:给imageNamed方法提供功能,每次加载图片就判断下图片是否加载成功。
    // 步骤一:先搞个分类,定义一个能加载图片并且能打印的方法+ (instancetype)imageWithName:(NSString *)name;
    // 步骤二:交换imageNamed和imageWithName的实现,就能调用imageWithName,间接调用imageWithName的实现。
    UIImage *image = [UIImage imageNamed:@"123"];
    }

    扩展

    @implementation UIImage (Image)
    // 加载分类到内存的时候调用
    +(void)load
    {
    // 交换方法
    // 获取imageWithName方法地址
    Method imageWithName = class_getClassMethod(self, @selector(imageWithName:));
    // 获取imageWithName方法地址
    Method imageName = class_getClassMethod(self, @selector(imageNamed:));
    // 交换方法地址,相当于交换实现方式
    method_exchangeImplementations(imageWithName, imageName);
    }
    // 不能在分类中重写系统方法imageNamed,因为会把系统的功能给覆盖掉,而且分类中不能调用super.
    // 既能加载图片又能打印
    +(instancetype)imageWithName:(NSString *)name
    {
    // 这里调用imageWithName,相当于调用imageName
    UIImage *image = [self imageWithName:name];
    if (image == nil) {
    NSLog(@"加载空的图片");
    }
    return image;
    }

    2.动态添加方法

    • 开发使用场景:加载类到内存的时候也比较耗费资源,需要给每个方法生成映射表,可以使用动态给某个类,添加方法解决。(经典面试题:有没有使用performSelector,其实主要想问你有没有动态添加过方法。)

    @implementation ViewController

    • (void)viewDidLoad {
      [super viewDidLoad];
      // Do any additional setup after loading the view, typically from a nib.
      Person *p = [[Person alloc] init];
      // 默认person,没有实现eat方法,可以通过performSelector调用,但是会报错。
      // 动态添加方法就不会报错
      [p performSelector:@selector(eat)];
      }
      @end

    @implementation Person
    // void(*)()
    // 默认方法都有两个隐式参数,
    void eat(id self,SEL sel)
    {
    NSLog(@"%@ %@",self,NSStringFromSelector(sel));
    }
    // 当一个对象调用未实现的方法,会调用这个方法处理,并且会把对应的方法列表传过来.
    // 刚好可以用来判断,未实现的方法是不是我们想要动态添加的方法
    +(BOOL)resolveInstanceMethod:(SEL)sel
    {
    if (sel == @selector(eat)) {
    // 动态添加eat方法
    // 第一个参数:给哪个类添加方法
    // 第二个参数:添加方法的方法编号
    // 第三个参数:添加方法的函数实现(函数地址)
    // 第四个参数:函数的类型,(返回值+参数类型) v:void @:对象->self :表示SEL->_cmd
    class_addMethod(self, @selector(eat), eat, "v@:");
    }
    return [super resolveInstanceMethod:sel];
    }
    @end

    3.给分类添加属性

    • 原理:给一个类声明属性,其实本质就是给这个类添加关联,并不是直接把这个值的内存空间添加到类存空间。

    @implementation ViewController
    -(void)viewDidLoad {
    [super viewDidLoad];
    // Do any additional setup after loading the view, typically from a nib.
    // 给系统NSObject类动态添加属性name
    NSObject *objc = [[NSObject alloc] init];
    objc.name = @"小码哥";

    NSLog(@"%@",objc.name);
    

    }
    @end


    // 定义关联的key
    static const char *key = "name";
    @implementation NSObject (Property)

    • (NSString *)name
      {
      // 根据关联的key,获取关联的值。
      return objc_getAssociatedObject(self, key);
      }
    • (void)setName:(NSString *)name
      {
      // 第一个参数:给哪个对象添加关联
      // 第二个参数:关联的key,通过这个key获取
      // 第三个参数:关联的value
      // 第四个参数:关联的策略
      objc_setAssociatedObject(self,key,name,OBJC_ASSOCIATION_RETAIN_NONATOMIC);
      }
      @end

    4.字典转模型

    • 自动根据一个字典,生成对应的属性,和字典中的key一一对应。

    @implementation NSObject (Log)
    // 自动打印属性字符串
    +(void)resolveDict:(NSDictionary *)dict{
    // 拼接属性字符串代码
    NSMutableString *strM = [NSMutableString string];
    // 1.遍历字典,把字典中的所有key取出来,生成对应的属性代码
    [dict enumerateKeysAndObjectsUsingBlock:^(id _Nonnull key, id _Nonnull obj, BOOL * _Nonnull stop) {
    //类型经常变,抽出来
    NSString *type;
    if ([obj isKindOfClass:NSClassFromString(@"__NSCFString")]) {
    type = @"NSString";
    }else if ([obj isKindOfClass:NSClassFromString(@"__NSCFArray")]){
    type = @"NSArray";
    }else if ([obj isKindOfClass:NSClassFromString(@"__NSCFNumber")]){
    type = @"int";
    }else if ([obj isKindOfClass:NSClassFromString(@"__NSCFDictionary")]){
    type = @"NSDictionary";
    }
    // 属性字符串
    NSString *str;
    if ([type containsString:@"NS"]) {
    str = [NSString stringWithFormat:@"@property (nonatomic, strong) %@ *%@;",type,key];
    }else{
    str = [NSString stringWithFormat:@"@property (nonatomic, assign) %@ %@;",type,key];
    }
    // 每生成属性字符串,就自动换行。
    [strM appendFormat:@"\n%@\n",str];
    }];
    // 把拼接好的字符串打印出来,就好了。
    NSLog(@"%@",strM);
    }
    @end

    相关文章

      网友评论

      本文标题:iOS runtime从菜鸟到应用(小灰进阶篇)

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