全网整合营销服务商

电脑端+手机端+微信端=数据同步管理

免费咨询热线:400-708-3566

IOS App 无代码入侵的方法hook详细介绍

iOS App 无代码入侵的方法hook

继续Objective-C runtime的研究

最近公司项目在做用户行为分析

于是App端在某些页面切换,交互操作的时候需要给统计系统发送一条消息

在几十个Controller 的项目里,一个一个地加代码那完全是不可能的,维护起来也是吃力

但这里需要处理的是 Controller, 可以有以下方式实现上述需求

1. 利用Objective-C 中的对象继承

  继承 在面向对象开发中是非常常用的,像我们现在做的项目工程中都会有一个BaseViewController,

所有新建的ViewController都继承BaseViewController,通过往BaseViewController中添加一些公共方法\属性 可以被他们的子类所调用

这是统一我们工程中所有视图控制器样式的一个主要途径

2.利用Category 和Runtime实行方法hook

  hook方案有一个好处,就是可以避免代码入侵,做到更加广泛的通用性.通过swizzling我们可以将原method与自己加入的method相结合,

即不需要在原有工程中加入代码,又能做到全局覆盖

两种方案对比:

  通过继承父类来实现 相对于hook来说 是较为准确的,因为需要被统计的页面都是继承于这个父类的控制器,而其他的如UINavigationController,系统自带的UIAlertController等则不会误入统计数据当中

  上面提到 hook方案是通过hook UIViewController viewdidload/viewdidappear等方法,而这些方法实际上 每个Controller 都会调用,那么就会出现不该出现的Controller 也出现在这里(如上面说到的UINavigationController和UIAlertController).但hook方案一个比较好的特点是无代码入侵,在不修改项目代码的前提下完成工作.

考虑到 行为分析统计系统 有可能被公司其他项目中所应用,这里采用hook方案.那么当中必然会出现 不该统计的却被统计 的情况,后面再作分析.

既然用到hook方案,又要用runtime 的swizzling

首先 新建一个UIViewController 的category

实现swizzling代码

+ (void)load{
  [super load];
  
  static dispatch_once_t onceToken;
  dispatch_once(&onceToken, ^{
    // 假如要打开controller的统计 ,则把下面这行代码打开
    __gbh_tracer_swizzleMethod([self class], @selector(viewDidAppear:), @selector(__gbh_tracer_viewDidAppear:));
  });
}

嗯,看到这里大家会发现 这里调用的是一个C的方法,然而这个C方法是怎么实现的呢?看下面

void __gbh_tracer_swizzleMethod(Class class, SEL originalSelector, SEL swizzledSelector){
  Method originalMethod = class_getInstanceMethod(class, originalSelector);
  Method swizzledMethod = class_getInstanceMethod(class, swizzledSelector);
  
  BOOL didAddMethod =
  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 ([filter containsObject:className]) return ; //如果该Controller在不允许发送log的列表里,则不能继续往下走
  
  if ([self.title isKindOfClass:[NSString class]] && self.title.length > 0){ //有标题的才符合我的要求
    // 这里发送log
  }

}

嗯,刚刚说到有部分Controller我是不发数据的,这里有两重判断,一个是加入到黑名单,另一个是 判断Controller的title属性是否为空

以上判断基本能满足我这个行为分析统计系统的需求,若还需要什么判断还可以继续加

以此 我只需要往工程里面添加这个Category,这个viewDidAppear就会被hook出来,可以为所欲为..

另外 需求中还提到 需要在应用启动的时候发送一次init消息

hook?可以,但我更倾向与利用category+NSNotification,因为系统中已经有 UIApplicationDidFinishLaunchingNotification

这种通知,直接用就可以

@implementation UIApplication (GBHTracer)
+ (void)load{
  [super load];
  
  static dispatch_once_t onceToken;
  dispatch_once(&onceToken, ^{ //只执行一次就可以了
    [[NSNotificationCenter defaultCenter]addObserver:self selector:@selector(__gbh_tracer_applicationDidFinishLaunching:) name:UIApplicationDidFinishLaunchingNotification object:nil];
  });
}

+ (void)__gbh_tracer_applicationDidFinishLaunching:(NSNotification *)noti{
  //应用启动时为所欲为!
}

@end

嗯..我们的行为分析统计系统就在原工程不Import一个头文件 不调用 任何一个方法就可以达到统计效果.

但是像什么操作响应的时候的统计,还是需要各位看官在响应中调用相应的方法

感谢阅读,希望能帮助到大家,谢谢大家对本站的支持!


# IOS  # App  # 无代码入侵的方法hook  # 入侵方法hook详解  # hook  # iOS开发中实现hook消息机制的方法探究  # iOS内存错误EXC_BAD_ACCESS的解决方法  # iOS开发中ViewController的页面跳转和弹出模态  # js和html5实现手机端刮刮卡抽奖效果完美兼容android/IOS  # iOS毛玻璃效果的实现及图片模糊效果的三种方法  # iOS开发中WebView的基本使用方法简介  # IOS开发代码分享之设置UISearchBar的背景颜色  # IOS获取各种文件目录路径的方法  # iOS开发中实现显示gif图片的方法  # 的是  # 就会  # 说到  # 为所欲为  # 就可以  # 都是  # 这是  # 我是  # 他们的  # 也有  # 会有  # 就在  # 还可以  # 有可能  # 子类  # 就不  # 我只  # 则是  # 是怎么  # 两种 


相关文章: 如何在香港服务器上快速搭建免备案网站?  婚礼视频制作网站,学习*后期制作的网站有哪些?  电商网站制作多少钱一个,电子商务公司的网站制作费用计入什么科目?  GML (Geography Markup Language)是什么,它如何用XML来表示地理空间信息?  免费网站制作模板下载,除了易企秀之外还有什么H5平台可以制作H5长页面,最好是免费的?  高防服务器租用指南:配置选择与快速部署攻略  建站之星代理如何优化在线客服效率?  ,南京靠谱的征婚网站?  已有域名如何快速搭建专属网站?  邀请函制作网站有哪些,有没有做年会邀请函的网站啊?在线制作,模板很多的那种?  常州自助建站费用包含哪些项目?  如何通过建站之星自助学习解决操作问题?  制作网站的软件下载免费,今日头条开宝箱老是需要下载怎么回事?  建站主机是什么?如何选择适合的建站主机?  全景视频制作网站有哪些,全景图怎么做成网页?  如何制作一个表白网站视频,关于勇敢表白的小标题?  官网网站制作腾讯审核要多久,联想路由器newifi官网  如何快速搭建高效WAP手机网站?  东莞市网站制作公司有哪些,东莞找工作用什么网站好?  javascript中的try catch异常捕获机制用法分析  如何续费美橙建站之星域名及服务?  ,怎么在广州志愿者网站注册?  网站制作和推广的区别,想自己建立一个网站做推广,有什么快捷方法马上做好一个网站?  昆明高端网站制作公司,昆明公租房申请网上登录入口?  建站之星安装后如何自定义网站颜色与字体?  在线制作视频的网站有哪些,电脑如何制作视频短片?  如何快速登录WAP自助建站平台?  ,制作一个手机app网站要多少钱?  C++用Dijkstra(迪杰斯特拉)算法求最短路径  公司门户网站制作公司有哪些,怎样使用wordpress制作一个企业网站?  如何选择可靠的免备案建站服务器?  javascript基本数据类型及类型检测常用方法小结  网站规划与制作是什么,电子商务网站系统规划的内容及步骤是什么?  如何在Golang中引入测试模块_Golang测试包导入与使用实践  如何正确选择百度移动适配建站域名?  建站主机选哪家性价比最高?  义乌企业网站制作公司,请问义乌比较好的批发小商品的网站是什么?  C++中引用和指针有什么区别?(代码说明)  教育培训网站制作流程,请问edu教育网站的域名怎么申请?  怎么制作一个起泡网,水泡粪全漏粪育肥舍冬季氨气超过25ppm,可以有哪些措施降低舍内氨气水平?  如何用AWS免费套餐快速搭建高效网站?  如何登录建站主机?访问步骤全解析  历史网站制作软件,华为如何找回被删除的网站?  建站之星如何实现网站加密操作?  上海网站制作网站建设公司,建筑电工证网上查询系统入口?  盘锦网站制作公司,盘锦大洼有多少5G网站?  建站之星如何助力企业快速打造五合一网站?  c++ stringstream用法详解_c++字符串与数字转换利器  实现点击下箭头变上箭头来回切换的两种方法【推荐】  如何通过FTP空间快速搭建安全高效网站? 

您的项目需求

*请认真填写需求信息,我们会在24小时内与您取得联系。