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

写在前面

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

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

但是要注意的是:OC是动态性语言,他的对象类型和真正要调用的方法是在运行时才确定的,所以这就决定了在oc中没有绝对的私有变量和私有方法的,通过runtime我们可以动态的去对类中所有的变量和方法进行操作,关于runtime的一些东西在
《iOS-RunTime介绍及使用 //www.greatytc.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函数的指针。
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 转至元数据结尾创建: 董潇伟,最新修改于: 十二月 23, 2016 转至元数据起始第一章:isa和Class一....
    40c0490e5268阅读 5,857评论 0 9
  • 我们常常会听说 Objective-C 是一门动态语言,那么这个「动态」表现在哪呢?我想最主要的表现就是 Obje...
    Ethan_Struggle阅读 6,595评论 0 7
  • 这篇文章完全是基于南峰子老师博客的转载 这篇文章完全是基于南峰子老师博客的转载 这篇文章完全是基于南峰子老师博客的...
    西木阅读 30,721评论 33 466
  • 对于从事 iOS 开发人员来说,所有的人都会答出【runtime 是运行时】什么情况下用runtime?大部分人能...
    梦夜繁星阅读 9,087评论 7 64
  • 今天发传单的时候, 发给了一位左手拿着好多个娃娃右手牵着女朋友的帅哥 我递给他 他说: 不用了,谢谢。 就此走过 ...
    木棉白杨阅读 1,352评论 0 0