runtime簡(jiǎn)稱(chēng)運(yùn)行時(shí)拢切。OC是運(yùn)行時(shí)機(jī)制,也就是在運(yùn)行時(shí)才做一些處理。例如:C語(yǔ)言在編譯的時(shí)候就知道要調(diào)用哪個(gè)方法函數(shù)袭祟,而OC在編譯的時(shí)候并不知道要調(diào)用哪個(gè)方法函數(shù),只有在運(yùn)行的時(shí)候才知道調(diào)用的方法函數(shù)名稱(chēng)捞附,來(lái)找到對(duì)應(yīng)的方法函數(shù)進(jìn)行調(diào)用巾乳。
導(dǎo)入
想要使用runtime,就要先導(dǎo)入runtime庫(kù)
一般導(dǎo)入message.h鸟召,因?yàn)閙essage.h包含了objc.h和runtime.h
#import <objc/message.h>
runtime作用
一:發(fā)送消息
方法調(diào)用的本質(zhì)就是放對(duì)象發(fā)送消息
/* new 會(huì)調(diào)用 init方法 */
People *man = [People new];
People *man = [[People alloc] init];
//屬性方法調(diào)用的方式
[man eat];
//類(lèi)方法調(diào)用方式
[People eat];
[[People class] eat];
//還有一種不常用的調(diào)用方式
[對(duì)象/類(lèi) performSelector:@selector(eat)];
//底層實(shí)現(xiàn)
objc_msgSend(對(duì)象/屬性, @selector(eat));
最終代碼查看方法:clang -rewrite-objc main.m
消息發(fā)送底層實(shí)現(xiàn) Build Setting設(shè)置msg為NO Xcode5之后使用runtime機(jī)制
OC 與 C 的對(duì)應(yīng)方法
[People class] == objc_getClass("People")
@selector() == sel_registerName()
二:交換方法
當(dāng)系統(tǒng)自帶的方法功能不夠胆绊,可以給系統(tǒng)自帶的方法擴(kuò)展一些功能,并保持原有的功能
例如我想知道當(dāng)前的URL是否為空如果每次都判斷一下的話(huà)會(huì)很麻煩欧募,如果我創(chuàng)建擴(kuò)展來(lái)寫(xiě)压状,又不知道內(nèi)部是如何實(shí)現(xiàn)的.
一:使用繼承來(lái)實(shí)現(xiàn)
//.h文件內(nèi)容
#import <Foundation/Foundation.h>
@interface CFURL : NSURL
+(instancetype)CFURLWithString:(NSString *)string;
@end
-----------------------------------------------
//.m文件內(nèi)容
#import "CFURL.h"
@implementation CFURL
+(instancetype)CFURLWithString:(NSString *)string{
CFURL *url = [super URLWithString:string];
if (url == nil) {
NSLog(@"url為空");
}
return url;
}
@end
二:使用runtime交換方法
//.h文件內(nèi)容
#import <Foundation/Foundation.h>
@interface NSURL (url)
+(instancetype)CF_URLWithStr:(NSString *)URLString;
@end
-----------------------------------------------
//.m文件內(nèi)容
#import "NSURL+url.h"
#import <objc/runtime.h>
@implementation NSURL (url)
+(void)load{
//最早的方法,比main還早
NSLog(@"load");
//1.拿到兩個(gè)Method
//2.進(jìn)行方法交換
Method m1 = class_getClassMethod([NSURL class], @selector(URLWithString:));
Method m2 = class_getClassMethod([NSURL class], @selector(CF_URLWithStr:));
//利用runtime進(jìn)行方法的交換
method_exchangeImplementations(m1, m2);
}
+(instancetype)CF_URLWithStr:(NSString *)URLString{
//交換了兩個(gè)方法
NSURL *url = [NSURL CF_URLWithStr:URLString];//注意這里不能再調(diào)用系統(tǒng)的方法
if (!url) {
NSLog(@"url為空");
}
return url;
}
@end
三:動(dòng)態(tài)添加方法
//.m文件
#import "People.h"
#import <objc/runtime.h>
@implementation People
//當(dāng)類(lèi)調(diào)用一個(gè)沒(méi)有實(shí)現(xiàn)的類(lèi)方法就會(huì)到這里8獭种冬!
+(BOOL)resolveClassMethod:(SEL)sel{
NSLog(@"%@",NSStringFromSelector(sel));
return [super resolveClassMethod:sel];
}
//當(dāng)類(lèi)調(diào)用一個(gè)沒(méi)有實(shí)現(xiàn)的對(duì)象方法就會(huì)到這里!舔糖!
+ (BOOL)resolveInstanceMethod:(SEL)sel
{
if (sel == @selector(eat)) {
// 動(dòng)態(tài)添加eat方法
/*
第一個(gè)參數(shù):給哪個(gè)類(lèi)添加方法
第二個(gè)參數(shù):添加方法的方法編號(hào)
第三個(gè)參數(shù):添加方法的函數(shù)實(shí)現(xiàn)(函數(shù)地址)
第四個(gè)參數(shù):函數(shù)的類(lèi)型娱两,(返回值+參數(shù)類(lèi)型) v:void @:對(duì)象->self :表示SEL->_cmd
*/
class_addMethod(self, @selector(eat), eat, "v@:");
}
return [super resolveInstanceMethod:sel];
}
// 默認(rèn)方法都有兩個(gè)隱式參數(shù),
void eat(id self,SEL sel)
{
NSLog(@"%@ %@",self,NSStringFromSelector(sel));
}
@end
四: 給分類(lèi)添加屬性
//.h文件
#import "NSObject.h"
@interface NSObject (Property)
//@property在分類(lèi)中只會(huì)生成set金吗、get方法的聲明 不會(huì)生成實(shí)現(xiàn)十兢,也不會(huì)生成_成員屬性
@property (nonatomic,copy)NSString name;
@end
-----------------------------------------------
//.m文件
// 定義關(guān)聯(lián)的key
static const char *key = "name";
@implementation NSObject (Property)
- (NSString *)name
{
// 根據(jù)關(guān)聯(lián)的key趣竣,獲取關(guān)聯(lián)的值。
return objc_getAssociatedObject(self, key);
}
- (void)setName:(NSString *)name
{
/*
第一個(gè)參數(shù):給哪個(gè)對(duì)象添加關(guān)聯(lián)
第二個(gè)參數(shù):關(guān)聯(lián)的key纪挎,通過(guò)這個(gè)key獲取
第三個(gè)參數(shù):關(guān)聯(lián)的value
第四個(gè)參數(shù):關(guān)聯(lián)的策略
*/
objc_setAssociatedObject(self, key, name, OBJC_ASSOCIATION_RETAIN_NONATOMIC);
}
@end