Runtime 介绍
这不是一遍介绍关于Runtime实现细节的文章,而是怎么利用Objective-C提供的Runtime API进行开发的文章!
Objective-C拥有相当多的动态特性,这些特性在运行程序时候发挥作用.
Objctive-C Runtime是个运行时的库,由C和汇编实现。通过Runtime封装的C结构体和函数可以在程序运行时创建、检查和修改类以及对象及其方法,甚至可以替换或交换方法的实现。
1)消息机制
在OOP术语中,消息传递是指一种在对象之间发送和接收消息的通信模式。
在Objective-C中,消息传递用于在调用类和类实例的方法,即接收者接收需要执行的消息。
使用案例
1
2
3
4
5
6
7
8
9
10
11
12
13
14
|
// 通过类名获取类
Class catClass = objc_getClass( "Cat" );
//注意Class实际上也是对象,所以同样能够接受消息,向Class发送alloc消息
Cat *cat = objc_msgSend(catClass, @selector(alloc));
//发送init消息给Cat实例cat
cat = objc_msgSend(cat, @selector(init));
//发送eat消息给cat,即调用eat方法
objc_msgSend(cat, @selector(eat));
//汇总消息传递过程
objc_msgSend(objc_msgSend(objc_msgSend(objc_getClass( "Cat" ), sel_registerName( "alloc" )), sel_registerName( "init" )), sel_registerName( "eat" ));
|
2)方法交换 Method Swizzling
Objective-C 提供了一下API用于动态替换类方法或者实例方法的实现:
- class_replaceMethod 替换类方法的定义
- method_exchangeImplementations 交换两个方法的实现(具体使用案例如下)
- method_setImplementation 设置一个方法的实现
注:class_replaceMethod 试图替换一个不存在的方法时候,会调用class_addMethod为该类增加一个新方法
使用案例
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
|
//Cat.m
+ ( void )load{
Method eatMethod = class_getInstanceMethod(self, @selector(eat));
Method shirtMethod = class_getInstanceMethod(self, @selector(shirt));
method_exchangeImplementations(eatMethod, shirtMethod);
}
- ( void )eat{
NSLog(@ "cat eat...." );
}
- ( void )shirt{
NSLog(@ "cat shirt...." );
}
|
3)动态加载方法
当调用一个未实现的方法,或者说发送未知的消息给接收者时候,消息的接受者会调用resolveInstanceMethod
使用案例
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
|
// Cat.m
//An Objective-C method is simply a C function that take at least two arguments—self and _cmd.
void run(id self, SEL _cmd, NSNumber *number){
NSLog(@ "run for %@" , number);
}
//收到run:消息时候,为该类添加一个方法实现
+ ( BOOL )resolveInstanceMethod:(SEL)sel{
if (sel == NSSelectorFromString(@ "run:" )){
class_addMethod(self, @selector(run:), run, "v@:@" );
return YES;
}
return [super resolveInstanceMethod:sel];
}
//另外针对类方法的为 resolveClassMethod
|
4)消息转发
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
|
// 第一步,消息接收者没有找到对应的方法时候,会先调用此方法,可在此方法实现中动态添加新的方法
// 返回YES表示相应selector的实现已经被找到,或者添加新方法到了类中,否则返回NO
+ ( BOOL )resolveInstanceMethod:(SEL)sel {
return YES;
}
// 第二步, 如果第一步的返回NO或者直接返回了YES而没有添加方法,该方法被调用
// 在这个方法中,我们可以指定一个可以返回一个可以响应该方法的对象, 注意如果返回self就会死循环
- (id)forwardingTargetForSelector:(SEL)aSelector {
return nil;
}
// 第三步, 如果forwardingTargetForSelector:返回了nil,则该方法会被调用,系统会询问我们要一个合法的『类型编码(Type Encoding)』
// 若返回 nil,则不会进入下一步,而是无法处理消息
- (NSMethodSignature *)methodSignatureForSelector:(SEL)aSelector {
return [NSMethodSignature signatureWithObjCTypes: "v@:" ];
}
// 当实现了此方法后,-doesNotRecognizeSelector: 将不会被调用
// 在这里进行消息转发
- ( void )forwardInvocation:(NSInvocation *)anInvocation {
// 在这里可以改变方法选择器
[anInvocation setSelector:@selector(unknown)];
// 改变方法选择器后,需要指定消息的接收者
[anInvocation invokeWithTarget:self];
}
- ( void )unknown {
NSLog(@ "unkown method......." );
}
// 如果没有实现消息转发 forwardInvocation 则调用此方法
- ( void )doesNotRecognizeSelector:(SEL)aSelector {
NSLog(@ "unresolved method :%@" , NSStringFromSelector(aSelector));
}
|
注: 『类型编码(Type Encoding)』
5)动态关联属性
对象在内存中的排布可以看成一个结构体,该结构体的大小并不能动态变化,所以无法在运行时动态给对象增加成员变量。相对的,对象的方法定义都保存在类的可变区域中。
如下图所示为Class 的描述信息,其中methodList为可访问类中定义的方法的指针的指针,通过修改该指针所指向的指针的值,我们可以实现为类动态增加方法实现。
因此,我们可以实现动态为一个类增加成员方法,但是却不能直接为类增加成员变量,这就是category的实现原理。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
|
//<objc/runtime.h>
struct objc_class {
Class isa OBJC_ISA_AVAILABILITY;
#if !__OBJC2__
Class super_class OBJC2_UNAVAILABLE;
const char *name OBJC2_UNAVAILABLE;
long version OBJC2_UNAVAILABLE;
long info OBJC2_UNAVAILABLE;
long instance_size OBJC2_UNAVAILABLE;
struct objc_ivar_list *ivars OBJC2_UNAVAILABLE;
struct objc_method_list **methodLists OBJC2_UNAVAILABLE;
struct objc_cache *cache OBJC2_UNAVAILABLE;
struct objc_protocol_list *protocols OBJC2_UNAVAILABLE;
#endif
} OBJC2_UNAVAILABLE;
|
使用案例
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
|
//Cat+Extend.h
@interface Cat (extend)
@property(nonatomic, copy) NSString *name;
@end
//Cat+Extend.m
@implementation Cat (extend)
- ( void )setName:(NSString *)name{
objc_setAssociatedObject(self, "name" , name, OBJC_ASSOCIATION_RETAIN_NONATOMIC);
}
- (NSString *)name{
return objc_getAssociatedObject(self, "name" );
}
@end
|
6)字典转模型应用
通过Class的结构体内容,可以看到ivars指针指向包含了类中成员变量的结构体,通过它可以得到类中定义的成员变量,而Objective-C中提供了相应的API方法: class_copyIvarList
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
|
//<objc/runtime.h>
struct objc_class {
Class isa OBJC_ISA_AVAILABILITY;
#if !__OBJC2__
Class super_class OBJC2_UNAVAILABLE;
const char *name OBJC2_UNAVAILABLE;
long version OBJC2_UNAVAILABLE;
long info OBJC2_UNAVAILABLE;
long instance_size OBJC2_UNAVAILABLE;
struct objc_ivar_list *ivars OBJC2_UNAVAILABLE;
struct objc_method_list **methodLists OBJC2_UNAVAILABLE;
struct objc_cache *cache OBJC2_UNAVAILABLE;
struct objc_protocol_list *protocols OBJC2_UNAVAILABLE;
#endif
} OBJC2_UNAVAILABLE;
|
使用案例
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
|
//Cat.h
@property(nonatomic, copy) NSString *cid;
@property(nonatomic, copy) NSString *age;
+ (instancetype)modelWithDict:(NSDictionary *)dict;
//Cat.m
+ (instancetype)modelWithDict:(NSDictionary *)dict{
id model = [[self alloc] init];
unsigned int count = 0;
Ivar *ivars = class_copyIvarList(self, &count);
for ( int i = 0 ; i < count; i++) {
Ivar ivar = ivars[i];
NSString *ivarName = [NSString stringWithUTF8String:ivar_getName(ivar)];
//这里注意,拿到的成员变量名为_cid,_age
ivarName = [ivarName substringFromIndex:1];
id value = dict[ivarName];
[model setValue:value forKeyPath:ivarName];
}
return model;
}
|
总结
以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,如果有疑问大家可以留言交流,谢谢大家对快网idc的支持。
相关文章
- 64M VPS建站:怎样优化以提高网站加载速度? 2025-06-10
- 64M VPS建站:是否适合初学者操作和管理? 2025-06-10
- ASP.NET自助建站系统中的用户注册和登录功能定制方法 2025-06-10
- ASP.NET自助建站系统的域名绑定与解析教程 2025-06-10
- 个人服务器网站搭建:如何选择合适的服务器提供商? 2025-06-10
- 2025-07-10 怎样使用阿里云的安全工具进行服务器漏洞扫描和修复?
- 2025-07-10 怎样使用命令行工具优化Linux云服务器的Ping性能?
- 2025-07-10 怎样使用Xshell连接华为云服务器,实现高效远程管理?
- 2025-07-10 怎样利用云服务器D盘搭建稳定、高效的网站托管环境?
- 2025-07-10 怎样使用阿里云的安全组功能来增强服务器防火墙的安全性?
快网idc优惠网
QQ交流群
-
2025-05-29 71
-
2025-05-29 29
-
2025-05-29 89
-
2025-06-04 78
-
2025-05-25 54