- ubuntu12.04环境下使用kvm ioctl接口实现最简单的虚拟机
- Ubuntu 通过无线网络安装Ubuntu Server启动系统后连接无线网络的方法
- 在Ubuntu上搭建网桥的方法
- ubuntu 虚拟机上网方式及相关配置详解
CFSDN坚持开源创造价值,我们致力于搭建一个资源共享平台,让每一个IT人在这里找到属于你的精彩世界.
这篇CFSDN的博客文章iOS开发中实现hook消息机制的方法探究由作者收集整理,如果你对这篇文章有兴趣,记得点赞哟.
method swizzling 原理 。
在objective-c中调用一个方法,其实是向一个对象发送消息,查找消息的唯一依据是selector的名字。利用objective-c的动态特性,可以实现在运行时偷换selector对应的方法实现,达到给方法挂钩的目的。 每个类都有一个方法列表,存放着selector的名字和方法实现的映射关系。imp有点类似函数指针,指向具体的method实现。 我们可以利用 method_exchangeimplementations 来交换2个方法中的imp, 我们可以利用 class_replacemethod 来修改类, 我们可以利用 method_setimplementation 来直接设置某个方法的imp, …… 归根结底,都是偷换了selector的imp,如下图所示: method swizzling 实践 。
举个例子好了,我想钩一下nsarray的lastobject 方法,只需两个步骤。 第一步:给nsarray加一个我自己的lastobject 。
。
。
。
乍一看,这不递归了么?别忘记这是我们准备调换imp的selector,[self mylastobject] 将会执行真的 [self lastobject] .
。
第二步:调换imp 。
。
。
。
控制台输出log:
。
。
结果很让人欣喜,是不是忍不住想给uiwebview的loadrequest: 加 todo 了呢?
示例 。
有了这个原理,接下来让我们来看一个实例: 下面先直接上源码: 。
。
// // testhookobject.m // testhookmessage // // created by maplecao on 13-2-28. // copyright (c) 2013年 maplecao. all rights reserved. // 。
。
#import "testhookobject.h" #import <objc/objc.h> #import <objc/runtime.h> 。
@implementation testhookobject 。
// this method will just excute once + (void)initialize { // 获取到uiwindow中sendevent对应的method method sendevent = class_getinstancemethod([uiwindow class], @selector(sendevent:)); method sendeventmyself = class_getinstancemethod([self class], @selector(sendeventhooked:)); // 将目标函数的原实现绑定到sendeventoriginalimplemention方法上 imp sendeventimp = method_getimplementation(sendevent); class_addmethod([uiwindow class], @selector(sendeventoriginal:), sendeventimp, method_gettypeencoding(sendevent)); // 然后用我们自己的函数的实现,替换目标函数对应的实现 imp sendeventmyselfimp = method_getimplementation(sendeventmyself); class_replacemethod([uiwindow class], @selector(sendevent:), sendeventmyselfimp, method_gettypeencoding(sendevent)); } 。
/* * 截获到window的sendevent * 我们可以先处理完以后,再继续调用正常处理流程 */ - (void)sendeventhooked:(uievent *)event { // do something what ever you want nslog(@"haha, this is my self sendeventmethod!!!!!!!"); // invoke original implemention [self performselector:@selector(sendeventoriginal:) withobject:event]; } 。
@end 。
。
。
下面我们来逐行分析一下上面的代码: 首先我们来看19行,这一行主要目的是获取到uiwindow原生的sendevent的method(一个结构体,用来对方法进行描述),接着第20行是获取到我们自己定义的类中的sendevent的method(这两个方法的签名必须一样,否则运行时报错)。第23行我们通过uiwindow原生的sendevent的method获取到对应的imp(一个函数指针),第24行使用运行时api class_addmethod给uiwindow类添加了一个叫sendeventoriginal的方法,该方法使用uiwindow原生的sendevent的实现,并且有着相同的方法签名(必须相同,否则运行时报错)。27行是获取我们自定义类中的sendeventmyself的imp,28行是关键的一行,这一行的主要目的是为uiwindow原生的sendevent指定一个新的实现,我们看到我们将该实现指定到了我们自己定义的sendeventmyself上。到了这儿我们就完成了偷梁换柱,大功告成。 执行上面这些行以后,我们就成功的将uiwindow的sendevent重定向到了我们自己的写的sendeventmyself的实现,然后将其原本的实现重定向到了我们给它新添加的方法sendeventoriginal中。而sendeventmyself中,我们首先可以对这个消息进行我们想要的处理,然后再通过41行调用sendeventoriginal方法转到正常的执行流程。 这块儿你可能有个困惑 “我们自定义类中明明是没有sendeventoriginal方法的啊?” 为什么执行起来不报错,而且还会正常执行?因为sendeventmyself是uiwindow的sendevent重定向过来的,所以在运行时该方法中的self代表的就是uiwindow的实例,而不再是testhookobject的实例了。加上sendeventoriginal是我们通过运行时添加到uiwindow的实例方法,所以可以正常调用。当然如果直接通过下面这种方式调用也是可以的,只不过编译器会提示警告(编译器没那么智能),因此我们采用了performselector的调用方式。 。
以上就是hook的实现,使用时我们只需要让testhookobject类执行一次初始话操作就可以了,执行完以后。uiwindow的sendevent消息就会会hook到我们的sendeventmyself中了。 下面是调用代码: 。
。
。
。
。
- (bool)application:(uiapplication *)application didfinishlaunchingwithoptions:(nsdictionary *)launchoptions { self.window = [[[uiwindow alloc] initwithframe:[[uiscreen mainscreen] bounds]] autorelease]; // override point for customization after application launch. self.viewcontroller = [[[testhookviewcontroller alloc] initwithnibname:@"testhookviewcontroller" bundle:nil] autorelease]; self.window.rootviewcontroller = self.viewcontroller; [self.window makekeyandvisible]; //hook uiwindow‘s sendevent method testhookobject *hooksendevent = [[testhookobject alloc] init]; [hooksendevent release]; uibutton *btn = [[uibutton alloc] initwithframe:cgrectmake(0, 0, 100, 100)]; btn.center = cgpointmake(160, 240); btn.backgroundcolor = [uicolor redcolor]; [btn addtarget:self action:@selector(btnaction:) forcontrolevents:uicontroleventallevents]; [self.window addsubview:btn]; [btn release]; return yes; } 。
。
。
代码中我们还专门添加了一个button来验证,hook完以后消息是否正常传递。经验证消息流转完全正常.
最后此篇关于iOS开发中实现hook消息机制的方法探究的文章就讲到这里了,如果你想了解更多关于iOS开发中实现hook消息机制的方法探究的内容请搜索CFSDN的文章或继续浏览相关文章,希望大家以后支持我的博客! 。
我需要一些说明。我可以直接写入 /dev/port 以直接访问并行端口并且它工作正常(我可以打开插入端口连接器的 LED)。但是,我想我可以用 /dev/mem 做同样的事情? (http://tld
按照目前的情况,这个问题不适合我们的问答形式。我们希望答案得到事实、引用或专业知识的支持,但这个问题可能会引发辩论、争论、投票或扩展讨论。如果您觉得这个问题可以改进并可能重新打开,visit the
我使用 Visual C++ 和 Win32 API 学习了 Windows 编程。如今,似乎大多数应用程序都是使用 C# 在 .NET 中开发的。我知道大多数时候 native 代码和托管代码之间没
请耐心等待。我正在制作一个 java 控制台,类似于此处找到的 DragonConsole https://code.google.com/p/dragonconsole/ 。一切都按计划进行,但我想
关闭。这个问题需要更多 focused .它目前不接受答案。 想要改进这个问题吗? 更新问题,使其只关注一个问题 editing this post . 关闭5年前。 Improve this que
Django 的开发服务器表现得很奇怪。访问它的浏览器在加载时卡住,任何退出它的尝试都不起作用。当我点击 control c看似相当,但实际上仍在运行。让它退出的唯一方法是重新启动我的电脑,这很令人沮
我正在使用 Flash Develop,并且创建了一个 ActionScript 3.0 项目。它启动并读取一个 xml 文件,其中包含图像的 url。我已将 url 保留在与 swf 相同的文件夹中
是否可以根据其 website 上提供的规范开发 AUTOSAR BSW 堆栈(例如用于 CAN 通信)?不购买任何昂贵的供应商工具?可以遵循哪些步骤?我被要求探索这种可能性。 最佳答案 是和否。工具
有人知道如何用音频文件的内容覆盖 iPhone 麦克风吗? 想象一个场景,您正在通话,并且想要播放一些简短的音频让其他人听到。 因此,有必要将麦克风(硬件)置于保持状态,并使用委托(delegate)
我遇到了这个问题,我的应用程序出现 EXC_BAD_ACCESS 错误并卡住/停止。我使用模拟器的“向左旋转”和“向右旋转”选项来模拟方向变化行为。导致此错误的可能原因有哪些?由于我没有获得有关错误的
我有超过 1 台 Mac,我想在所有这些 Mac 上进行开发。我知道我需要在每台机器上同步我的手机,但这是我遇到的最小的问题。看起来我无法在手机上运行应用程序,除了在其中之一上开发的应用程序。 是否有
在手机上测试时,我的应用程序在特定点崩溃。控制台显示此消息 Tue Jan 27 15:47:14 unknown SpringBoard[22] : Application com.myprof.
我有一个案例,我从服务器获取信息。我的应用程序有一个选项卡栏和导航按钮。我希望应用程序显示进度指示器并禁用所有其他控件,以便用户在从服务器提取数据时无法跳转。我怎样才能实现这个目标? 我想到的一种方法
有时,当我尝试“构建”/编译下载的源代码时,我会收到以下警告: ld: warning: directory '/Volumes/Skiiing2/CD/ViewBased/Unknown Path/
我无法在 Apple 文档中找到关于开发和分发配置之间差异的明确解释。我目前正在使用开发配置在我的 iPhone 上进行开发和测试。我打算将该应用程序分发到我的 Beta 测试中,我想知道: 我需要使
我在使用 SharePoint 时遇到的最大挑战之一是它不能很好地适应典型的项目环境,其中至少包含开发和生产环境。我遇到的最多的问题是内容和列表是如此紧密地耦合在一起,以至于如果不在生产环境中执行内容
我失败了fist step让 Eclipse(对我来说是全新的)为 ARM 开发做好准备。 我在 Windows 10 中安装了 Eclipse。我想我应该安装 xpm,但我不知道在哪里输入此命令:
首先,我告诉你-我是编码新手 我正在使用vs代码来学习c++,它不会产生像dev c++或codeblocks这样的调试器。我看了一些视频,其中我们必须编辑json文件,这对于初学者来说非常复杂。有人
我失败了fist step让 Eclipse(对我来说是全新的)为 ARM 开发做好准备。 我在 Windows 10 中安装了 Eclipse。我想我应该安装 xpm,但我不知道在哪里输入此命令:
我开发了一个 Ionic 应用程序(iOS 和 Android 的混合)。我有 Xcode 8.3.3 并购买了一年的 Apple Developer Program 订阅。 我不想测试我的应用并将其
我是一名优秀的程序员,十分优秀!