使用运行时runtime来调用私有方法

一、先了解下runtime RunTime简称运行时。就是系统在运行的时候的一些机制,其中最主要的是消息机制。对于C语言,函数的调用在编译的时候会决定调用哪个函数。编译完成之后直接顺序执行,无任何二义性。OC的函数调用称为消息发送。属于动态调用过程。在编译的时候并不能决定真正调用哪个函数(事实证明,在编 译阶段,OC可以调用任何函数,即使这个函数并未实现,只要申明过就不会报错。而C语言在编译阶段就会报错)。只有在真正运行的时候才会根据函数的名称找 到对应的函数来调用。
【使用运行时runtime来调用私有方法】在OC中有些文件只给出了声明的头文件,里面有一些可以被我们调用的函数和属性,而头文件中没有声明的我们便望尘莫及了,不可能被我们调用到。而根据runtime的特点,我们可以看到这些私有方法和属性,并且调用。(虽然调用私有方法的app上架会被reject,但是仍然阻挡不住我们的脚步)

//Calculator.h#import @interface Calculator : NSObject- (NSNumber *)sumAddend1:(NSNumber *)adder1 addend2:(NSNumber *)adder2; - (NSNumber *)sumAddend1:(NSNumber *)adder1 :(NSNumber *)adder2; @end

//Calculator.m#import #import "Calculator.h"@implementation Calculator- (NSNumber *)sumAddend1:(NSNumber *)adder1 addend2:(NSNumber *)adder2{ NSLog(@"Invorking method on %@ object with selector %@",[self class],NSStringFromSelector(_cmd)); return [NSNumber numberWithInteger:([adder1 integerValue] + [adder2 integerValue])]; }- (NSNumber *)sumAddend1:(NSNumber *)adder1 :(NSNumber *)adder2{NSLog(@"Invorking method on %@ object with selector %@",[self class],NSStringFromSelector(_cmd)); return [NSNumber numberWithInteger:([adder1 integerValue] + [adder2 integerValue])]; }@end

//main文件 #import #import "Calculator.h"int main(int argc, const char * argv[]) { @autoreleasepool {// 创建类的一个实例(对象) Calculator *calc = [Calculator new]; NSNumber *addend1 = [NSNumber numberWithInteger:25]; NSNumber *addend2 = [NSNumber numberWithInteger:10]; NSNumber *addend3 = [NSNumber numberWithInteger:15]; // 选择器使用@selector()指令,在编译时创建的 SEL selector1 = @selector(sumAddend1:addend2:); id sum1 = [calc performSelector:selector1 withObject:addend1 withObject:addend2]; NSLog(@"Sum of %@ + %@ = %@",addend1,addend2,sum1); // 选择器使用NSSelectorFromString()函数,在程序运行时创建的 SEL selector2 = NSSelectorFromString(@"sumAddend1::"); id sum2 = [calc performSelector:selector2 withObject:addend1 withObject:addend3]; NSLog(@"Sum of %@ + %@ = %@",addend1,addend3,sum2); } return 0; }

运行结果(ps:需要引入头文件:,才可以调用runtime里面的函数):
2016-11-18 21:10:55.147 7.2.3.****使用对象消息****[1059:46233] Invorking method on Calculator object with selector sumAddend1:addend2: 2016-11-18 21:10:55.148 7.2.3.****使用对象消息****[1059:46233] Sum of 25 + 10 = 35 2016-11-18 21:10:55.148 7.2.3.****使用对象消息****[1059:46233] Invorking method on Calculator object with selector sumAddend1:: 2016-11-18 21:10:55.148 7.2.3.****使用对象消息****[1059:46233] Sum of 25 + 15 = 40 Program ended with exit code: 0

可以看到,通过runtime机制,在程序运行时才决定去调用哪个函数。在main中,使用运行时调用了Calculator类中的两个对象方法。并且这两个方法还在头文件中声明了,一定会被外界调用到。但是运行时机制可以在运行时去决定调用哪一个函数,那么它一定能知道类中有哪些方法,无论是私有的还是公有的。
根据这一点,把Calculator类头文件中声明的函数注释掉,在程序运行时应该还是能调用到这两个方法。
//Calculator.h#import @interface Calculator : NSObject//- (NSNumber *)sumAddend1:(NSNumber *)adder1 addend2:(NSNumber *)adder2; //- (NSNumber *)sumAddend1:(NSNumber *)adder1 :(NSNumber *)adder2; @end

运行:
编译没有报错,并且得到的结果和之前的是一模一样的。证明猜想正确。
二、调用私有方法 现在来试试调用私有方法:
首先拿UIView下手:
先来看看这个类下面所有的函数
//先获取类名 NSString *className = NSStringFromClass([UIView class]); const char *cClassName = [className UTF8String]; id theClass = objc_getClass(cClassName); //用来计数 unsigned int outCount; Method *m =class_copyMethodList(theClass,&outCount); NSLog(@"%d",outCount); for (int i = 0; i

结果:
**206-11-18 21:24:22.402 JavaScriptCore-Demo[1088:52065] 1278** **2016-11-18 21:24:22.402 JavaScriptCore-Demo[1088:52065] _accessibilityCirclePathBasedOnBoundsWidth** **2016-11-18 21:24:22.403 JavaScriptCore-Demo[1088:52065] _accessibilityOnlyComparesByXAxis** **2016-11-18 21:24:22.403 JavaScriptCore-Demo[1088:52065] _accessibilitySortedElementsWithin** **2016-11-18 21:24:22.403 JavaScriptCore-Demo[1088:52065] _accessibilityViewController** **2016-11-18 21:24:22.403 JavaScriptCore-Demo[1088:52065] _accessibilityVerticalSizeClass** **2016-11-18 21:24:22.403 JavaScriptCore-Demo[1088:52065] _accessibilityHorizontalSizeClass** **2016-11-18 21:24:22.404 JavaScriptCore-Demo[1088:52065] _accessibilityAutomaticIdentifier** **2016-11-18 21:24:22.404 JavaScriptCore-Demo[1088:52065] _accessibilitySupportGesturesAttributes** **2016-11-18 21:24:22.404 JavaScriptCore-Demo[1088:52065] _accessibilityIsUserInteractionEnabled** **2016-11-18 21:24:22.404 JavaScriptCore-Demo[1088:52065] _accessibilityServesAsContainingParentForOrdering** **2016-11-18 21:24:22.404 JavaScriptCore-Demo[1088:52065] _accessibilityUserTestingChildren** **2016-11-18 21:24:22.405 JavaScriptCore-Demo[1088:52065] _accessibilityObscuredScreenAllowedViews** **2016-11-18 21:24:22.405 JavaScriptCore-Demo[1088:52065] accessibilityIsWindow** **2016-11-18 21:24:22.410 JavaScriptCore-Demo[1088:52065] _accessibilityHitTest:withEvent:** **2016-11-18 21:24:22.410 JavaScriptCore-Demo[1088:52065] _axResponderChain** **2016-11-18 21:24:22.411 JavaScriptCore-Demo[1088:52065] _accessibilityChildVendingParent** **2016-11-18 21:24:22.411 JavaScriptCore-Demo[1088:52065] _superAccessibilityHitTest:withEvent:** **2016-11-18 21:24:22.411 JavaScriptCore-Demo[1088:52065] _accessibilityPostNotification:** **2016-11-18 21:24:22.411 JavaScriptCore-Demo[1088:52065] _axWindowSubviews** **2016-11-18 21:24:22.411 JavaScriptCore-Demo[1088:52065] _axPrintSubviews:string:** **2016-11-18 21:24:22.412 JavaScriptCore-Demo[1088:52065] _accessibilityShouldHitTestLayers** **2016-11-18 21:24:22.412 JavaScriptCore-Demo[1088:52065] _accessibilityCheckForAllowedModalView:event:** **2016-11-18 21:24:22.412 JavaScriptCore-Demo[1088:52065] _accessibilityMaxFuzzyHitTestDistance** **2016-11-18 21:24:22.412 JavaScriptCore-Demo[1088:52065] _accessibilityModalViewBlocksView:blockerView:**

UIView类中总共有1278个方法,我这只展示了一部分,大家可以亲自试一试。
在这些方法中我挑选了一个setMaskView:方法,想调用下这个私有方法。
使用运行时runtime来调用私有方法
文章图片
Snip20161118_1.png 这是我没有调用前的界面
UIView *maskView = [[UIView alloc] initWithFrame:CGRectMake(0, 0, 200, 200)]; maskView.backgroundColor = [UIColor redColor]; SEL selector = NSSelectorFromString(@"setMaskView:"); [self.view performSelector:selector withObject:maskView];

使用运行时runtime来调用私有方法
文章图片
Snip20161118_2.png 这是我调用后的界面。
到此,我们就成功的调用了私有方法。令我疑惑的是,我给maskView的框架是(0, 0, 200, 200),背景颜色是红色,最后显示出来的效果是这个样子,可能是我理解上还存在一些问题,希望大家能够指正批评。

    推荐阅读