使用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”
第一次写文章,哪里写的不好,有问题请多多指出。