UIViewController_Swizzled的原理和使用

实现的功能:
  可用于跟踪复杂程序的一个非常简单的类库源码,查看每个viewController的层级嵌套顺序,显示一个UIViewController的时候日志记录一下。
原理:
  置换 UIViewController 的 viewDidAppear: 方法
  Objecitve-C的重要特性是Runtime(运行时),在Interacting with the Runtime(交互运行)中,运行时函数部分,苹果给出了/usr/lib/libobjc.A.dylib库,这个共享库提供支持动态属性的objective - c语言,通过其接口,可以用于开发将其他语言运行于Objective-C上的中间层(桥接层),库里的函数定义为纯C语言。
  其中#import

 <objc/runtime.h>

  运行时确认函数用于动态调用,代码中:

+(void)load
{
    isSwizzed = NO;
}

  一般情况下,类别里的方法会重写掉主类里相同命名的方法。如果有两个类别实现了相同命名的方法,只有一个方法会被调用。但 +load: 是个特例,当一个类被读到内存的时候, runtime 会给这个类及它的每一个类别都发送一个 +load: 消息。

-(void)swizzviewDidAppear:(BOOL)animated
{
    [self printPath];
    // Call the original method (viewWillAppear)
    [self swizzviewDidAppear:animated];
}

  代码看起来可能有点奇怪,像递归不是么。当然不会是递归,因为在 runtime 的时候,函数实现已经被交换了。调用 viewDidAppear: 会调用你实现的 swizzled_viewDidAppear:,而在 swizzled_viewDidAppear: 里调用 swizzled_viewDidAppear: 实际上调用的是原来的 viewDidAppear: 所以效果就是viewDidAppear的调用会先执行swizzviewDidAppear,再执行swizzviewDidAppear里的swizzviewDidAppear(也就是原来viewDidAppear里实现的代码)

  函数替换

  • class_getInstanceMethod
  • class_addMethod
  • class_replaceMethod

使用步骤:
  1、从github下载UIViewController-Swizzled,可以cocopad,也可以直接add进项目。
  2、直接在.pch文件#import ,这样所有的controller都包含了这个头文件。
  3、在AppDelegate.m的didFinishLaunchingWithOptions方法中,调用宏SWIZZ_IT_WITH_TAG(LOGTAG)或者SWIZZ_IT。
  

- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions {
    // Override point for customization after application launch.
    SWIZZ_IT_WITH_TAG(LOGTAG);
    return YES;
}

源码地址:https://github.com/RuiAAPeres/UIViewController-Swizzled

Objective-C 的运行时中最具争议的黑魔法:method swizzling:http://www.cocoachina.com/ios/20160121/15076.html

你可能感兴趣的:(IOS原生开发)