IOS App 无代码入侵的方法hook详细介绍
发布时间 - 2026-01-10 21:53:36 点击率:次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图片的方法
# 的是
# 就会
# 说到
# 为所欲为
# 就可以
# 都是
# 这是
# 我是
# 他们的
# 也有
# 会有
# 就在
# 还可以
# 有可能
# 子类
# 就不
# 我只
# 则是
# 是怎么
# 两种
相关栏目:
【
网站优化151355 】
【
网络推广146373 】
【
网络技术251813 】
【
AI营销90571 】
相关推荐:
Laravel Seeder填充数据教程_Laravel模型工厂Factory使用
Laravel怎么判断请求类型_Laravel Request isMethod用法
php 三元运算符实例详细介绍
东莞专业网站制作公司有哪些,东莞招聘网站哪个好?
百度输入法ai组件怎么删除 百度输入法ai组件移除工具
Laravel如何使用.env文件管理环境变量?(最佳实践)
如何在云虚拟主机上快速搭建个人网站?
智能起名网站制作软件有哪些,制作logo的软件?
Laravel如何实现URL美化Slug功能_Laravel使用eloquent-sluggable生成别名【方法】
大连网站制作费用,大连新青年网站,五年四班里的视频怎样下载啊?
Windows11怎样设置电源计划_Windows11电源计划调整攻略【指南】
Laravel策略(Policy)如何控制权限_Laravel Gates与Policies实现用户授权
如何在建站之星绑定自定义域名?
海南网站制作公司有哪些,海口网是哪家的?
Laravel如何自定义错误页面(404, 500)?(代码示例)
Python正则表达式进阶教程_复杂匹配与分组替换解析
软银砸40亿美元收购DigitalBridge 强化AI资料中心布局
Win11怎么关闭资讯和兴趣_Windows11任务栏设置隐藏小组件
如何在阿里云服务器自主搭建网站?
如何自定义建站之星网站的导航菜单样式?
如何制作新型网站程序文件,新型止水鱼鳞网要拆除吗?
如何在阿里云虚拟主机上快速搭建个人网站?
bootstrap日历插件datetimepicker使用方法
Laravel如何实现事件和监听器?(Event & Listener实战)
Laravel怎么使用Blade模板引擎_Laravel模板继承与Component组件复用【手册】
创业网站制作流程,创业网站可靠吗?
合肥制作网站的公司有哪些,合肥聚美网络科技有限公司介绍?
如何在阿里云高效完成企业建站全流程?
Laravel Octane如何提升性能_使用Laravel Octane加速你的应用
Laravel模型关联查询教程_Laravel Eloquent一对多关联写法
Laravel如何实现本地化和多语言支持_Laravel多语言配置与翻译文件管理
Android 常见的图片加载框架详细介绍
LinuxShell函数封装方法_脚本复用设计思路【教程】
Android okhttputils现在进度显示实例代码
Python制作简易注册登录系统
Windows10怎样连接蓝牙设备_Windows10蓝牙连接步骤【教程】
Laravel Debugbar怎么安装_Laravel调试工具栏配置指南
php增删改查怎么学_零基础入门php数据库操作必知基础【教程】
黑客入侵网站服务器的常见手法有哪些?
Python文件异常处理策略_健壮性说明【指导】
JavaScript如何实现类型判断_typeof和instanceof有什么区别
Laravel Telescope怎么调试_使用Laravel Telescope进行应用监控与调试
Laravel PHP版本要求一览_Laravel各版本环境要求对照
Laravel如何保护应用免受CSRF攻击?(原理和示例)
Laravel如何与Inertia.js和Vue/React构建现代单页应用
如何在万网ECS上快速搭建专属网站?
高性能网站服务器配置指南:安全稳定与高效建站核心方案
深圳网站制作平台,深圳市做网站好的公司有哪些?
北京网站制作费用多少,建立一个公司网站的费用.有哪些部分,分别要多少钱?
Laravel如何配置中间件Middleware_Laravel自定义中间件拦截请求与权限校验【步骤】

