什么是runtime
runtime就是运行时,是系统在运行时的一些动态机制,它是一套底层的API,我们平时编写的OC代码,最终会转换为runtime实现。
runtime的作用
- 可以利用runtime获取一个类的属性列表,可以通过runtime拿到一个类的所有成员属性。
首先要导入runtime.h的头文件
#import <objc/runtime.h>
1 // 定义一个无符号整型数字(用于记录类中的成员变量的个数)
2 unsigned int count = 0;
3
4 // 使用这个runtime函数,可以获取一个类的成员变量,返回一个ivar类型的数组
5 Ivar *ivars = class_copyIvarList([Person class], &count);
6 NSLog(@"%d",count); // count打印出来为3,即(person类的成员变量个数)
7 for (int i = 0;i < count;i++) {
8 // 获取每个成员变量
9 Ivar memmber = ivars[i];
10 const char *name = ivar_getName(memmber);
11 NSLog(@"%s",name);
12 }
打印结果为:
2016-01-22 08:43:37.292 c语言函数等[2528:16328] _age
2016-01-22 08:43:37.293 c语言函数等[2528:16328] _name
2016-01-22 08:43:37.293 c语言函数等[2528:16328] _height
,第5行class_copyIvarList([Person class], &count)
- runtime可以动态的给一个类增加方法,也可以交换自定义的方法和系统的方法的实现。
imageNamed: 这个方法添加一些额外的功能,这时候可以写一个方法来替换系统的 imageNamed:
1 #import <objc/runtime.h>
2
3 @implementation UIImage (Extension)
4 /**
5 * 只要分类被装载到内存中,就会调用1次
6 */
7 + (void)load
8 {
9 Method otherMehtod = class_getClassMethod(self, @selector(imageWithName:));
10 Method originMehtod = class_getClassMethod(self, @selector(imageNamed:));
11 // 交换2个方法的实现
12 method_exchangeImplementations(otherMehtod, originMehtod);
13 }
可以在UIImage的分类中,使用这个runtime的 method_exchangeImplementations(otherMehtod, originMehtod) 函数,将系统的方法 imageNamed: 和自定义的方法 imageWithName: 交换实现,之后当你调用这个方法 imageNamed:时,就会使用自定义方法 imageWithName:
什么是runloop
- 其实它内部就是do-while循环,在这个循环内部不断地处理各种任务(比如Source、Timer、Observer),能让线程不被系统终止
- 一个线程对应一个RunLoop,主线程的RunLoop默认已经启动,子线程的RunLoop得手动启动(调用run方法)
- RunLoop只能选择一个Mode启动,如果当前Mode中没有任何Source(Sources0、Sources1)、Timer,那么就直接退出RunLoop
runloop的作用
- 可以使用runloop,在一个子线程中长期监控某个事件
1 - (void)viewDidLoad {
2 [super viewDidLoad];
3 // 开启一个线程让它执行run方法,如果run方法过了,线程就会死掉
4 self.thread = [[NSThread alloc] initWithTarget:self selector:@selector(run) object:nil];
5 [self.thread start];
6 }
7 - (void)run
8 {
9 NSLog(@"run----%@",[NSThread currentThread]);
10 // 给线程执行的方法添加运行时,添加source1 和timer,让这个进程不死,并在没有任务时进入休眠状态
11 [[NSRunLoop currentRunLoop] addPort:[NSPort port] forMode:NSDefaultRunLoopMode];
12 [[NSRunLoop currentRunLoop] run];
13 // 可以在这里添加要循环执行的代码
14 }
上面的代码如果不加runloop,run方法执行完,self.thread这个线程就会被释放
- 可以让某些事件(行为、任务)在特定模式下执行(例如轮播广告,在用户拖动图片的时候,不去循环轮播图片(不做定时任务),等用户松开手指时才开始计算循环周期(开始定时任务))
- runloop常用的三种模式:
- NSDefaultRunLoopMode:runloop默认的模式,程序启动模式主线程的runloop就是在NSDefaultRunLoopMode模式下运行的。
- UITrackingRunLoopMode:runloop在用户点击或触摸屏幕时,会自动切换到该模式。
- NSRunLoopCommonModes:指标记为common modes的所有模式,即前两者的集合。
1 - (void)timer
2 {
3 NSTimer *timer = [NSTimer timerWithTimeInterval:2.0 target:self selector:@selector(run) userInfo:nil repeats:YES];
4 // 定时器只运行在NSDefaultRunLoopMode下,一旦RunLoop进入其他模式,这个定时器就不会工作
5 [[NSRunLoop currentRunLoop] addTimer:timer forMode:NSDefaultRunLoopMode];
6 }
7
8 - (void)run
9 {
10 NSLog(@"----run");
11 }
如上代码,在 NSDefaultRunLoopMode模式下添加一个定时器任务(每2秒打印一次run这个方法),程序一启动后,是NSDefaultRunLoopMode,所以一直打印,当用户触摸屏幕上的UIScrollView时,此时自动切换到UITrackingRunLoopMode,定时器停止工作,当用户松开手时,定时器重新启动,run方法继续执行打印。如果将
如果将上面代码加入到NSRunLoopCommonModes,那么不管用户是否触摸UIScrollView,定时器都会正常工作。