线程之间通信
performSelector
- (void)performSelectorOnMainThread:(SEL)aSelector withObject:(id)arg waitUntilDone:(BOOL)wait; //返回主线程
- (void)performSelector:(SEL)aSelector onThread:(NSThread *)thr withObject:(id)arg waitUntilDone:(BOOL)wait; //返回指定线程
load和initialize方法分别在调用时机
+ load
- 它应该是在一个类进行装载的时候触发,就是不管这个类有没有被调用,只要它被装载,那么它就会运行这个方法,只要被添加到编译源下面就会执行。
- 不管子类有没有写load方法,父类的load都只会执行一次。
- load方法执行的时候,系统为脆弱状态,如果我们在load里面需要调用其它类的实例对象(或类对象)的属性或者方法,必须要确保那个依赖类的load方法执行完毕
- 当加载资源过大会造成性能问题(用户体验、系统抖动)
+ initialize
- 如果没有用到该类,就算加载完毕也不会执行该方法(这点与load方法不同,load方法是只要加载就执行,initialize方法必须是第一次使用该类的时候才触发且触发一次)
- 以懒加载的方式被调用的,不是启动程序就调用。
runloop和线程的关系
- 每条线程都有唯一的一个 RunLoop 对象与之对应的
- 主线程的 RunLoop 是自动创建并启动
- 子线程的 RunLoop 需要手动创建(懒加载,只创建一次)
- RunLoop保存在一个全局的Dictionary里,线程作为key,RunLoop作为value
- RunLoop在第一次获取时创建,在线程结束时销毁
1
2
3
4
5
6
7//启动RunLoop
[[NSRunLoop currentRunLoop] run];
//第一个参数:指定运行模式
//第二个参数:指定 RunLoop 的过期时间,即:到了这个时间后RunLoop 就失效了
[[NSRunLoop currentRunLoop] runMode:kCFRunLoopDefaultModebeforeDate:[NSDate distantFuture]];
block的实质是什么?__block修饰的变量为什么能在block里面能改变其值?
- block其实也是一个oc对象,内部有一个isa指针
- __block 所起到的作用就是只要观察到该变量被 block 所持有,就将“外部变量”在栈中的内存地址放到了堆中。进而在block内部也可以修改外部变量的值。
- block不允许修改外部变量的值,这里所说的外部变量的值,指的是栈中指针的内存地址。栈区是红灯区,堆区才是绿灯区。
创建线程的方法是什么?如果在主线程中执行代码,方法是什么? 如果想延时执行代码、方法又是什么?
创建线程
- NSThread创建
- GCD的dispatch创建
- 使用子类化的NSOperation,然后将其加入NSOperationQueue;
在主线程执行代码
- performSelectorOnMainThread 或者 GCD中 dispatch_main
延迟执行
- performSelectorOnMainThread:onThread:withObject:waitUntilDone: 或者 GCD中 dispatch_after
delegate和notification区别和场景
区别
- delegate效率高于notification
- delegate比notification更加直观
场景
- nofification适合2个不是很紧密的模块, 例如多线程传值。
- delegate是一种简单的回调,且主要集中在一个模块。例如想把值从底层传到上层。