美文网首页IOSiOS开发专题我的待读文章
iOS 利用runTime进行“私有方法”替换

iOS 利用runTime进行“私有方法”替换

作者: 劉光軍_MVP | 来源:发表于2017-11-28 17:08 被阅读132次

写在前面

首先声明:题目中所说的“私有方法”只是我们感官上的感觉,OC 中没有绝对的私有变量和私有方法。

关于私有变量和“私有方法”
  • 私有变量 用@private来声明私有变量,只允许本类访问。
  • “私有方法”OC中没有提供关键字来声明私有方法,可以通过category的匿名类Extension通过在一个只在类的.m文件中来声明一个只能被本类访问的方法。

但是要注意的是:OC是动态性语言,他的对象类型和真正要调用的方法是在运行时才确定的,所以这就决定了在oc中没有绝对的私有变量和私有方法的,通过runtime我们可以动态的去对类中所有的变量和方法进行操作,关于runtime的一些东西在
《iOS-RunTime介绍及使用 http://www.jianshu.com/p/5408dcab8f02》中介绍。比如通过class_copyIvarListclass_copyMethodList这两个方法。

正题

我们在项目中可能会遇到这样的情况,你引入了一个库,库里面的.m文件中的方法你不能修改,但是你又必须要用到,就像我们遇到的场景,我需要跳到库里的mainViewController中,在mainViewController有个navigationController的pop方法,在这个mainViewController中定义的pop方法是popToRootViewControllerAnimated但是我需要的是pop到指定viewController中,但是我们又不能修改mainViewController这个文件中的方法,该怎么办呢?这里说一个利用runtime解决的方法:

替换“私有方法”

在我们当前的VC中,我们生命一个方法changeLiveMethod用来执行替换方法操作。
代码如下:

  • 新的pop方法
- (void)popToLiveListVC1
{
    UIViewController *target = nil;
    for (UIViewController * controller in self.navigationController.viewControllers) { //遍历
        if ([controller isKindOfClass:[LiveListViewController class]]) { //这里判断是否为你想要跳转的页面
            target = controller;
        }
    }
    if (target)
    {
        [self.navigationController popToViewController:target animated:YES]; //跳转
    }
}
  • 替换方法
//当前pop方法1
    SEL thisSelector = @selector(popToLiveListVC1);
    Method thisMethod = class_getInstanceMethod([self class], thisSelector);
    
   //获取mainViewCOntroller 中的gotoViewController方法
//    unsigned int count = 0;
//    Method *membFuns = class_copyMethodList([MainViewController class], &count);
    NSString *mainPopMethodStr = @"gotoViewController";
//    for (int i =0 ; i < count; i++) {
//        SEL name = method_getName(membFuns[i]);
//        NSString *methodName = [NSString stringWithCString:sel_getName(name) encoding:NSUTF8StringEncoding];
//        NSLog(@"m method:%@",methodName);
//        if ([methodName isEqualToString:@"gotoViewController"]) {
//            mainPopMethodStr = methodName;
//        }
//    }
    SEL mainPopSelector = NSSelectorFromString(mainPopMethodStr);
    Method mainPopMethod = class_getInstanceMethod([MainViewController class], mainPopSelector);
    method_exchangeImplementations(thisMethod, mainPopMethod);

代码分析: 先取到当前我们需要的新方法 Method,然后获取库里面MainViewController私有方法,我们用Method mainPopMethod = class_getInstanceMethod([MainViewController class], @selector(gotoViewController));这个方法编译器会报个警告,所以我们用SEL的这个方式获取这个mainPopMethod ,然后进行method_exchangeImplementations方法的交换,这个方法就是改变类的分发表,使它在解析消息的时候,将一个选择器selector对应到另一个替换的实现,并且将该选择器对应的原始实现关联到新的选择器中。

对SEL Method IMP的一些说明
  • 选择器(Selector-typedef struct objc_selector *SEL ):用于在运行时表示一个方法的名称,一个方法选择器就是一个C字符串,在运行时会被注册或者映射,选择器是由编译器生成的,并在类被加载的时候由运行时自动进行映射。
  • 方法(Method-typedef struct objc_method *Method):在类的定义中代表一个方法的类型。
  • 实现(Implementation- typedef id (*IMP)(id, SEL, ...)):这是一个指向方法实现函数起始地址的指针,这个函数的第一个参数是指向self的指针,第二个参数是方法选择器,然后是方法的参数。
    理解它们之间关系的最好方式是:一个类维护着一张分发表(dispatch table),用来解析运行时发来的消息;该表的每个入口是一个方法(Method),其中的key就是选择器(SEL),对应一个实现(IMP),即一个指向底层c函数的指针。

相关文章

网友评论

    本文标题:iOS 利用runTime进行“私有方法”替换

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