什么是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

  1. 其实它内部就是do-while循环,在这个循环内部不断地处理各种任务(比如Source、Timer、Observer),能让线程不被系统终止
  2. 一个线程对应一个RunLoop,主线程的RunLoop默认已经启动,子线程的RunLoop得手动启动(调用run方法)
  3. 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常用的三种模式:
  1.  NSDefaultRunLoopMode:runloop默认的模式,程序启动模式主线程的runloop就是在NSDefaultRunLoopMode模式下运行的。
  2. UITrackingRunLoopMode:runloop在用户点击或触摸屏幕时,会自动切换到该模式。
  3. 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,定时器都会正常工作。