美文网首页
Runtime 替换方法(Method Swizzling)

Runtime 替换方法(Method Swizzling)

作者: 你猜我猜不猜你猜我猜不猜 | 来源:发表于2017-10-25 09:44 被阅读0次

使用Runtime Method Swizzling 替换UILabel初始化方法,实现修改默认字体

创建一个UILabel的Category

UILabel+ChangeFont.h
UILabel+ChangeFont.m

在.m内实现以下代码

+(void)load{
    //只执行一次这个方法
/*
object_getClass :
当obj为实例变量时,object_getClass(obj)与[obj class]输出结果一直,均获得isa指针,即指向类对象的指针。
当obj为类对象时,object_getClass(obj)返回类对象中的isa指针,即指向元类对象的指针;[obj class]返回的则是其本身
*/
        Class class = object_getClass(self);
        SEL originalSelector = @selector(init);
        SEL swizzledSelector = @selector(ChangeInit);
// 使用object_getClass 获取isa指针时即使是类方法时也可以使用class_getInstanceMethod获取方法的Method 因为类是元类的实例对象
        Method originalMethod = class_getInstanceMethod(class, originalSelector);
        Method swizzledMethod = class_getInstanceMethod(class, swizzledSelector);

        /*
         先判断目标类内部是否实现了这个方法,
         有两种情况要考虑一下:
            第一种情况是要替换的方法并没有在目标类中实现,而是在其父类中实现了;
            第二种情况是这个方法已经存在于目标类中

         对于第一种情况,因为class_getInstanceMethod 会返回父类的实现,如果直接替换,就会替换掉父类的实现,而不是目标类中的实现。
        
         举个具体的例子, 假设要替换掉-[NSView description].
         如果NSView 没有实现-description (可选的) 那你就可会得到NSObject的方法。如果调用method_exchangeImplementations , 你就会把NSObject 的方法替换成你的代码。
         */
        if (class_addMethod(class,
                            originalSelector,
                            method_getImplementation(swizzledMethod),
                            method_getTypeEncoding(swizzledMethod))) {
  /*
 如果是YES(方法原先不在目标类中,后来添加上去),
首先说明了本类中没有original method,我们在第一步中获取的original method 是在父类中的,所以我们自己在本类中创建了一个new original method,
现在new original method 与第一步获取的original method 没有任何的关系。那么现在 new original Method的IMP是swizzle IMP,
那么我现在调用class_replaceMethod可以把可能从父类中拿到的original method中的实现方法赋值给swizzle method,这样就完成了方法替换
*/
            class_replaceMethod(class,
                                swizzledSelector,
                                method_getImplementation(originalMethod),
                                method_getTypeEncoding(originalMethod));
           
        } else {
            // 如果NO,就是第二情况(方法已经存在于目标类中)。这时可以通过method_exchangeImplementations来完成交换:
            method_exchangeImplementations(originalMethod, swizzledMethod);
        }
}


/**
 *在这些方法中将你的字体名字换进去
 */

- (instancetype)ChangeInit
{
    // 外部调用init会被转化到ChangeInit
    // 这里会调用改变前的方法(init),不会产生死循环
    id __self = [self ChangeInit];
    UIFont * font = [UIFont fontWithName:@"Zapfino" size:self.font.pointSize];
    if (font) {
        self.font=font;
    }
    return __self;
}
@end

使用时不需要调用内部任何东西,毫无侵入

UILabel *label = [[UILabel alloc] init];

初始化出来的label字体就是“Zapfino”

第一次写文章,哪里写的不好,有问题请多多指出。

相关文章

网友评论

      本文标题:Runtime 替换方法(Method Swizzling)

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