利用Objective-C运行时hook函数的三种方法

方法一,hook已有公开头文件的类:

首先写一个Utility函数:

#import <objc/runtime.h>
void exchangeMethod(Class aClass, SEL oldSEL, SEL newSEL)
{
    Method oldMethod = class_getInstanceMethod(aClass, oldSEL);
    assert(oldMethod);
    Method newMethod = class_getInstanceMethod(aClass, newSEL);
    assert(newMethod);
    method_exchangeImplementations(oldMethod, newMethod);
}

现在,目标是hook UIWebView没公开的函数

- (void)webView:(id)arg1 didFinishLoadForFrame:(id)arg2;

因为已知类的声明,所以可以使用category:

@interface UIWebView (Hook)
+ (void)hook;
- (void)hook_webView:(id)arg1 didFinishLoadForFrame:(id)arg2;
@end
@implementation UIWebView (Hook)
+ (void)hook
{
    // hook UIWebView中表示一个HTML的frame加载完毕的函数
    exchangeMethod([UIWebView class],
                   @selector(webView:didFinishLoadForFrame:),
                   @selector(hook_webView:didFinishLoadForFrame:));
}
- (void)hook_webView:(id)arg1 didFinishLoadForFrame:(id)arg2
{
    // 因为交换了selector和implementation的映射,原样调一下本函数实际会调用被hook的函数。
    [self hook_webView:arg1 didFinishLoadForFrame:arg2];
    NSLog(@"webView:didFinishLoadForFrame:");
}

在程序启动的时候调用一下 [UIWebView hook] 即可。使用一个UIWebView打开一个网页,即会打印NSLog。

方法二,hook没有公开头文件的类,需要另建一个类作为新函数载体,然后先为被hook的类增加函数,再替换。
UIWebView体系中有一个类叫UIWebBrowserView,它是真正显示网页的UIView,并有部分函数是作为WebCore向外发送回调信息的接收者。
现我们去hook UIWebBrowserView的这个函数:

- (void)webView:(id)arg1 didFinishLoadForFrame:(id)arg2;

嗯,是的,这个函数和UIWebView的一样,实际上就是UIWebBrowserView会再调用通知到UIWebView的同名函数。
创建一个类,不要与被hook的类同名,例如加了个Hook前缀:

@interface UIWebBrowserViewHook : NSObject
+ (void)hook;
- (void)hook_webView:(id)arg1 didFinishLoadForFrame:(id)arg2;
@end

其中以hook_为前缀的新增函数的实现与方法一中相同,差别在类函数中:

@implementation UIWebBrowserViewHook
+ (void)hook
{
    Class aClass = objc_getClass("UIWebBrowserView");
    SEL sel = @selector(hook_webView:didFinishLoadForFrame:);
    // 为UIWebBrowserView增加函数
    class_addMethod(aClass, sel, class_getMethodImplementation([self class], sel), "v@:@@");
    // 交换实现
    exchangeMethod(aClass, @selector(webView:didFinishLoadForFrame:), sel);
}

在程序启动的时候调用一下 [UIWebBrowserViewHook hook] 即可。使用一个UIWebView打开一个网页,即会打印NSLog。

方法三,hook没有公开头文件的类,另建一个类作为新函数载体,用新函数替换旧函数,并把旧函数保存到静态变量里:

继续以UIWebBrowserView为例子。注意新函数可以与被hook的函数同名

@interface UIWebBrowserViewHook : NSObject
+ (void)hook;
- (void)webView:(id)arg1 didFinishLoadForFrame:(id)arg2;
@end

需要用到另一个Utility函数:

inline void replaceImplementation(Class newClass, Class hookedClass, SEL sel, IMP& oldImp)
{
    Method old = class_getInstanceMethod(hookedClass, sel);
    IMP newImp = class_getMethodImplementation(newClass, sel);
    oldImp = method_setImplementation(old, newImp);
}

当两个selector不同名时,以上函数再增加一个参数即可。

下面是实现:

@implementation UIWebBrowserViewHook
static IMP webView_didFinishLoadForFrame = NULL;
+ (void)hook
{
    Class hookedClass = objc_getClass("UIWebBrowserView");
    SEL sel = @selector(webView:didFinishLoadForFrame:);
    replaceImplementation([self class], hookedClass, sel, webView_didFinishLoadForFrame);
}

- (void)webView:(id)arg1 didFinishLoadForFrame:(id)arg2
{
    // 需要这样来调用被替换掉的原实现
    webView_didFinishLoadForFrame(self, @selector(webView:didFinishLoadForFrame:), arg1, arg2);
    NSLog(@"webView:didFinishLoadForFrame:");
}
@end

在程序启动的时候调用一下 [UIWebBrowserViewHook hook] 即可。使用一个UIWebView打开一个网页,即会打印NSLog。

三种方法的比较:

最方便的当然是第一种,但需要是hook有公开头文件的类。

方法二和方法三都新建了一个类,方法二需要描述selector的types,这个比较麻烦,如例子中的"v@:@@"表示返回值为void,第一和第二个参数都是id。方法三不用types,但要增加全局变量。

Objective-C的runtime参考资料:
http://developer.apple.com/library/ios/#documentation/Cocoa/Conceptual/ObjCRuntimeGuide/Introduction/Introduction.html#//apple_ref/doc/uid/TP40008048

时间: 2024-08-31 00:43:22

利用Objective-C运行时hook函数的三种方法的相关文章

JavaScript中定义函数的三种方法

 这篇文章主要介绍了JavaScript中定义函数的三种方法,本文直接给出代码实现,同时给出了构造函数的相关知识,需要的朋友可以参考下     在JavaScript的世界里,定义函数的方法多种多样,这正是JavaScript灵活性的体现,但是正是这个原因让初学者摸不着头脑,尤其对于没有 语言基础的同学.正所谓条条大道通罗马,但是如果道路太多,会让行路者不知所措,因为不知道走那条路才是正途,呵呵,废话一大篇,闲言少叙,先看代码: 代码如下: /*第一种方法,使用function语句,格式如下*/

JavaScript 申明函数的三种方法 每个函数就是一个对象(一)_javascript技巧

一.申明函数的三种方法 (Declaring a Function) 方法一:function functionName([parameters]){functionBody}; Example D1 复制代码 代码如下: function add(a, b) { return a+b; } alert(add(1,2)); // produces 3 当我们这样申明函数时,函数的内容被解释(但没有执行,要我们调用该函数才会执行),同时,一个名为add的对象已经建立. 方法二:将一个未命名的函数

JavaScript中定义函数的三种方法_javascript技巧

在JavaScript的世界里,定义函数的方法多种多样,这正是JavaScript灵活性的体现,但是正是这个原因让初学者摸不着头脑,尤其对于没有 语言基础的同学.正所谓条条大道通罗马,但是如果道路太多,会让行路者不知所措,因为不知道走那条路才是正途,呵呵,废话一大篇,闲言少叙,先看代码: 复制代码 代码如下: /*第一种方法,使用function语句,格式如下*/ function fn(){ alert("这是使用function语句进行函数定义"); } fn(); /*第二种方法

结构体类型数据作为函数参数(三种方法)_C 语言

(1)用结构体变量名作为参数. 复制代码 代码如下: #include<iostream>#include<string>using namespace std;struct Student{ string name; int score; };int main(){ Student one; void Print(Student one); one.name="千手"; one.score=99; Print(one); cout<<one.nam

当APP在后台运行时,接收到消息.有部分方法会执行两次.导致通知中显示收到的消息数是实际的两倍.

问题描述 当APP在后台运行时,接收到消息.有部分方法会执行两次.导致通知中显示收到的消息数是实际的两倍.求大神解答 解决方案 什么部分方法会执行两次,你是否监听了多次呢?解决方案二:我的也有这样的问题,求答案解决方案三:我也遇到了这个问题.我分析了一下,估计是下面的原因:测试手机使用的是小米,最新的SDK集成了小米的推送服务.导致在接受消息的时候,会接受小米的推送消息,然后环信的连接就连上了,环信又接受了一次.这就出现两条消息提示,但是实际只有一条消息.

js function定义函数的几种方法

 这篇文章主要介绍了js function定义函数的几种方法,需要的朋友可以参考下 js function定义函数的4种方法   1.最基本的作为一个本本分分的函数声明使用.   代码如下: function func(){}  或  var func=function(){};    2.作为一个类构造器使用:   代码如下: function class(){}  class.prototype={};  var item=new class();    3.作为闭包使用:   代码如下:

C#如何用三种方法对数组排序,请使用学过的知识,并且调用所写的函数

问题描述 C#如何用三种方法对数组排序,请使用学过的知识,并且调用所写的函数 C#如何用三种方法对数组排序,请使用学过的知识,并且调用所写的函数 解决方案 排序方法有很多,比如冒泡.快速.插入.选择.堆.归并.基数等等,你们数据结构应该学过. 自己google 排序算法名 C# 就有你要的程序 解决方案二: 天知道你学了什么知识. 你可以用linq的orderby.List的sort或者自己写一个排序算法实现: void mysort(int[] arr) { int temp = 0; for

JS中创建函数的三种方式及区别_基础知识

1.函数声明 function sum1(n1,n2){ return n1+n2; }; 2.函数表达式,又叫函数字面量 var sum2=function(n1,n2){ return n1+n2; }; 两者的区别:解析器会先读取函数声明,并使其在执行任何代码之前可以访问:而函数表达式则必须等到解析器执行到它所在的代码行才会真正被解释执行. 自执行函数严格来说也叫函数表达式,它主要用于创建一个新的作用域,在此作用域内声明的变量,不会和其它作用域内的变量冲突或混淆,大多是以匿名函数方式存在,

网页中JS函数自动执行常用三种方法_javascript技巧

本文为大家分享了在网页中JS函数自动执行常用方法,供大家参考,具体内容如下 一.JS方法 1.最简单的调用方式,直接写到html的body标签里面: <body onload="myfunction()"> <html> <body onload="func1();func2();func3();"> </body> </html> 2.在JS语句调用: <script type="text