IOS App 无代码入侵的方法hook详细介绍
iOSApp无代码入侵的方法hook
继续Objective-Cruntime的研究
最近公司项目在做用户行为分析
于是App端在某些页面切换,交互操作的时候需要给统计系统发送一条消息
在几十个Controller的项目里,一个一个地加代码那完全是不可能的,维护起来也是吃力
但这里需要处理的是Controller,可以有以下方式实现上述需求
1.利用Objective-C中的对象继承
继承在面向对象开发中是非常常用的,像我们现在做的项目工程中都会有一个BaseViewController,
所有新建的ViewController都继承BaseViewController,通过往BaseViewController中添加一些公共方法\属性可以被他们的子类所调用
这是统一我们工程中所有视图控制器样式的一个主要途径
2.利用Category和Runtime实行方法hook
hook方案有一个好处,就是可以避免代码入侵,做到更加广泛的通用性.通过swizzling我们可以将原method与自己加入的method相结合,
即不需要在原有工程中加入代码,又能做到全局覆盖
两种方案对比:
通过继承父类来实现相对于hook来说是较为准确的,因为需要被统计的页面都是继承于这个父类的控制器,而其他的如UINavigationController,系统自带的UIAlertController等则不会误入统计数据当中
上面提到hook方案是通过hookUIViewControllerviewdidload/viewdidappear等方法,而这些方法实际上每个Controller都会调用,那么就会出现不该出现的Controller也出现在这里(如上面说到的UINavigationController和UIAlertController).但hook方案一个比较好的特点是无代码入侵,在不修改项目代码的前提下完成工作.
考虑到行为分析统计系统有可能被公司其他项目中所应用,这里采用hook方案.那么当中必然会出现不该统计的却被统计的情况,后面再作分析.
既然用到hook方案,又要用runtime的swizzling
首先新建一个UIViewController的category
实现swizzling代码
+(void)load{ [superload]; staticdispatch_once_tonceToken; dispatch_once(&onceToken,^{ //假如要打开controller的统计,则把下面这行代码打开 __gbh_tracer_swizzleMethod([selfclass],@selector(viewDidAppear:),@selector(__gbh_tracer_viewDidAppear:)); }); }
嗯,看到这里大家会发现这里调用的是一个C的方法,然而这个C方法是怎么实现的呢?看下面
void__gbh_tracer_swizzleMethod(Classclass,SELoriginalSelector,SELswizzledSelector){ MethodoriginalMethod=class_getInstanceMethod(class,originalSelector); MethodswizzledMethod=class_getInstanceMethod(class,swizzledSelector); BOOLdidAddMethod= class_addMethod(class, originalSelector, method_getImplementation(swizzledMethod), method_getTypeEncoding(swizzledMethod)); if(didAddMethod){ class_replaceMethod(class, swizzledSelector, method_getImplementation(originalMethod), method_getTypeEncoding(originalMethod)); }else{ method_exchangeImplementations(originalMethod,swizzledMethod); } }
这是一个标准的swizzling写法,当然了github上面也有关于swizzling的开源库,用起来也顺手这里就不多说
看回第一块代码,红色的viewDidAppear是即将被我hook的方法,__gbh_tracer_viewDidAppear则是我需要实现的方法
-(void)__gbh_tracer_viewDidAppear:(BOOL)animated{ [self__gbh_tracer_viewDidAppear:animated];//由于方法已经被交换,这里调用的实际上是viewDidAppear:方法 //设置不允许发送数据的Controller NSArray*filter=@[@"UINavigationController",@"UITabBarController"]; NSString*className=NSStringFromClass(self.class); if([filtercontainsObject:className])return;//如果该Controller在不允许发送log的列表里,则不能继续往下走 if([self.titleisKindOfClass:[NSStringclass]]&&self.title.length>0){//有标题的才符合我的要求 //这里发送log } }
嗯,刚刚说到有部分Controller我是不发数据的,这里有两重判断,一个是加入到黑名单,另一个是判断Controller的title属性是否为空
以上判断基本能满足我这个行为分析统计系统的需求,若还需要什么判断还可以继续加
以此我只需要往工程里面添加这个Category,这个viewDidAppear就会被hook出来,可以为所欲为..
另外需求中还提到需要在应用启动的时候发送一次init消息
hook?可以,但我更倾向与利用category+NSNotification,因为系统中已经有UIApplicationDidFinishLaunchingNotification
这种通知,直接用就可以
@implementationUIApplication(GBHTracer) +(void)load{ [superload]; staticdispatch_once_tonceToken; dispatch_once(&onceToken,^{//只执行一次就可以了 [[NSNotificationCenterdefaultCenter]addObserver:selfselector:@selector(__gbh_tracer_applicationDidFinishLaunching:)name:UIApplicationDidFinishLaunchingNotificationobject:nil]; }); } +(void)__gbh_tracer_applicationDidFinishLaunching:(NSNotification*)noti{ //应用启动时为所欲为! } @end
嗯..我们的行为分析统计系统就在原工程不Import一个头文件不调用任何一个方法就可以达到统计效果.
但是像什么操作响应的时候的统计,还是需要各位看官在响应中调用相应的方法
感谢阅读,希望能帮助到大家,谢谢大家对本站的支持!