Objective-C中runtime机制的应用

一、初识runtime

Objective-C是一种动态语言,所谓动态语言,是在程序执行时动态的确定变量类型,执行变量类型对应的方法的。因此,在Object-C中常用字符串映射类的技巧来动态创建类对象。因为OC的动态语言特性,我们可以通过一些手段,在程序运行时动态的更改对象的变量甚至方法,这就是我们所说的runtime机制。

二、你还有什么办法操作这样的变量么?

首先,我们先来看一个例子,这里有我创建的一个MyObject类:

//.h===========================

@interface MyObject : NSObject

{

@private

int privateOne;

NSString * privateTow;;

}

@end

//=============================

//.m===========================

@interface MyObject()

{

@private

NSString * privateThree;

}

@end

@implementation MyObject

- (instancetype)init

{

self = [super init];

if (self) {

privateOne=1;

privateTow=@"Tow";

privateThree=@"Three";

}

return self;

}

-(NSString *)description{

return [NSString stringWithFormat:@"one=%d\ntow=%@\nthree=%@\n",privateOne,privateTow,privateThree];

}

@end

//=============================

这个类是相当的安全,首先,在头文件中没有提供任何的方法接口,我们没有办法使用点语法做任何操作,privateOne和PrivateTow两个变量虽然声明在了头文件中,却是私有类型的,通过指针的方式我们虽然可以看到他们,却不能做任何读取修改的操作,xcode中的提示如下:


他会告诉我们,这是一个私有的变量,我们不能使用。对于privateThree,我们更是束手无策,不仅不能使用,我们甚至都看不到它的存在。那么对于这种情况,你有什么办法操作这些变量么?对,是时候展现真正的技术了:runtime!

三、通过runtime获取对象的变量列表

要操作对象的变量,我们首先应该要捕获这些变量,让他们无处遁形。无论声明在头文件或是实现文件,无论类型是公开的还是私有的,只要声明了这个变量,系统就会为其分配空间,我们就可以通过runtime机制捕获到它,代码如下:

#import "ViewController.h"

#import "MyObject.h"

//包含runtime头文件

#import

@interface ViewController ()

@end

@implementation ViewController

- (void)viewDidLoad {

[super viewDidLoad];

//我们先声明一个unsigned int型的指针,并为其分配内存

unsigned int * count = malloc(sizeof(unsigned int));

//调用runtime的方法

//Ivar:方法返回的对象内容对象,这里将返回一个Ivar类型的指针

//class_copyIvarList方法可以捕获到类的所有变量,将变量的数量存在一个unsigned int的指针中

Ivar * mem = class_copyIvarList([MyObject class], count);

//进行遍历

for (int i=0; i< *count ; i++) {

//通过移动指针进行遍历

Ivar var = * (mem+i);

//获取变量的名称

const char * name = ivar_getName(var);

//获取变量的类型

const char * type = ivar_getTypeEncoding(var);

NSLog(@"%s:%s\n",name,type);

}

//释放内存

free(count);

//注意处理野指针

count=nil;

}

- (void)didReceiveMemoryWarning {

[super didReceiveMemoryWarning];

// Dispose of any resources that can be recreated.

}

@end

打印结果如下,其中i表示int型:


是不是小吃惊了一下,无论变量在哪里,只要它在,就让它无处遁形。

四、让我找到你,就让我改变你!

仅仅能够获得变量的类型和名字或许并没有什么卵用,没错,我们获取变量的目的不是为了观赏,而是为了操作它,这对runtime来说,也是小事一碟。代码如下:

- (void)viewDidLoad {

[super viewDidLoad];

//获取变量

unsigned int  count;

Ivar * mem = class_copyIvarList([MyObject class],&count);

//创建对象

MyObject * obj = [[MyObject alloc]init];

NSLog(@"before runtime operate:%@",obj);

//进行变量的设置

object_setIvar(obj, mem[0],10);

object_setIvar(obj, mem[1], @"isTow");

object_setIvar(obj, mem[2], @"isThree");

NSLog(@"after runtime operate:%@",obj);

}

Tip:在修改int型变量的时候,你或许会遇到一个问题,ARC下,编译器不允许你将int类型的值赋值给id,在buildset中将Objective-C Automatic Reference Counting修改为No即可。

打印效果如下:


可以看到,那些看似非常安全的变量被我们修改了。

五、让我看看你的方法吧

变量通过runtime机制我们可以取到和改变值,那么我们再大胆一点,试试那些私有的方法,首先我们在MyObject类中添加一些方法,我们只实现,并不声明他们:

@interface MyObject()

{

@private

NSString * privateThree;

}

@end

@implementation MyObject

- (instancetype)init

{

self = [super init];

if (self) {

privateOne=1;

privateTow=@"Tow";

privateThree=@"Three";

}

return self;

}

-(NSString *)description{

return [NSString stringWithFormat:@"one=%d\ntow=%@\nthree=%@\n",privateOne,privateTow,privateThree];

}

-(NSString *)method1{

return @"method1";

}

-(NSString *)method2{

return @"method2";

}

这样的方法我们在外面是无法调用他们的,和操作变量的思路一样,我们先要捕获这些方法:

//获取所有成员方法

Method * mem = class_copyMethodList([MyObject class], &count);

//遍历

for(int i=0;i

SEL name = method_getName(mem[i]);

NSString * method = [NSString stringWithCString:sel_getName(name) encoding:NSUTF8StringEncoding];

NSLog(@"%@\n",method);

}

打印如下:



得到了这些方法名,我们大胆的调用即可:

MyObject * obj = [[MyObject alloc]init];

NSLog(@"%@",[obj method1]);

Tip:这里编译器不会给我们方法提示,放心大胆的调用即可。

六、动态的为类添加方法

这个runtime机制最强大的部分要到了,试想,如果我们可以动态的向类中添加方法,那将是一件多么令人激动的事情,注意,这里是动态的添加,和类别的最大不同在于这种方式是运行时才决定是否添加方法的。

- (void)viewDidLoad {

[super viewDidLoad];

//添加一个新的方法,第三个参数是返回值的类型v是void,i是int,:是SEL,对象是@等

class_addMethod([MyObject class], @selector(method3), (IMP)logHAHA, "v");

unsigned int count = 0;

Method * mem = class_copyMethodList([MyObject class], &count);

for(int i=0;i

SEL name = method_getName(mem[i]);

NSString * method = [NSString stringWithCString:sel_getName(name) encoding:NSUTF8StringEncoding];

NSLog(@"%@\n",method);

}

MyObject * obj = [[MyObject alloc]init];

//运行这个方法

[obj performSelector:@selector(method3)];

}

//方法的实现

void logHAHA(){

NSLog(@"HAHA");

}

运行结果如下


从前五行可以看出,方法已经加进去了,从最后一行可以看出,执行没有问题。

七、做点小手脚

程序员总是得寸进尺的,现在,我们要做点事情,用我们的函数替换掉类中的函数:

- (void)viewDidLoad {

[super viewDidLoad];

MyObject * obj = [[MyObject alloc]init];

//替换之前的方法

NSLog(@"%@", [obj method1]);

//替换

class_replaceMethod([MyObject class], @selector(method1), (IMP)logHAHA, "v");

[obj method1];

}

void logHAHA(){

NSLog(@"HAHA");

}

打印如下:


这次够cool吧,通过这个方法,我们可以把系统的函数都搞乱套。当然,runtime还有许多很cool的方法:

id object_copy(id obj, size_t size)

拷贝一个对象

id object_dispose(id obj)

释放一个对象

const char *object_getClassName(id obj)

获取对象的类名

ive

void method_exchangeImplementations(Method m1, Method m2)

交换两个方法的实现

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,686评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,668评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,160评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,736评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,847评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,043评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,129评论 3 410
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,872评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,318评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,645评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,777评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,470评论 4 333
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,126评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,861评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,095评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,589评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,687评论 2 351

推荐阅读更多精彩内容

  • 转至元数据结尾创建: 董潇伟,最新修改于: 十二月 23, 2016 转至元数据起始第一章:isa和Class一....
    40c0490e5268阅读 1,692评论 0 9
  • 我们常常会听说 Objective-C 是一门动态语言,那么这个「动态」表现在哪呢?我想最主要的表现就是 Obje...
    Ethan_Struggle阅读 2,186评论 0 7
  • 原文出处:南峰子的技术博客 Objective-C语言是一门动态语言,它将很多静态语言在编译和链接时期做的事放到了...
    _烩面_阅读 1,223评论 1 5
  • Runtime是一套比较底层的纯C语言API,包含了很多底层的C语言API。在我们平时编写的OC代码中,程序运行时...
    这个年纪的情愫丶阅读 589评论 5 3
  • 文中的实验代码我放在了这个项目中。 以下内容是我通过整理[这篇博客] (http://yulingtianxia....
    茗涙阅读 914评论 0 6