RunTime 参考:https://www.jianshu.com/p/6ebda3cd8052
RunLoop 参考:https://www.jianshu.com/p/de752066d0ad
什么是Block,Block的本质是什么?
block本质上也是一个OC对象,它内部也有个isa指针
block是封装了函数调用以及函数调用环境的OC对象
block是封装函数及其上下文的OC对象
int age=10;
void (^Block)(void) = ^{
NSLog(@"age:%d",age);
};
age = 20;
Block();
输出值为 age:10
原因:创建block的时候,已经把age的值存储在里面了。
auto int age = 10;
static int num = 25;
void (^Block)(void) = ^{
NSLog(@"age:%d,num:%d",age,num);
};
age = 20;
num = 11;
Block();
输出结果为:age:10,num:11
愿意:auto变量block访问方式是值传递,static变量block访问方式是指针传递
为什么block对auto和static变量捕获有差异?
auto自动变量可能会销毁的,内存可能会消失,不采用指针访问;static变量一直保存在内存中,指针访问即可
为什么block对auto和static变量捕获有差异?
auto自动变量可能会销毁的,内存可能会消失,不采用指针访问;static变量一直保存在内存中,指针访问即可
block对全局变量的捕获方式是?
block不需要对全局变量捕获,都是直接采用取值的
block的类型,取决于isa指针,可以通过调用class方法或者isa指针查看具体类型,最终都是继承自NSBlock类型
__NSGlobalBlock __ ( _NSConcreteGlobalBlock )
__NSStackBlock __ ( _NSConcreteStackBlock )
__NSMallocBlock __ ( _NSConcreteMallocBlock )
各类型的block在内存中如何分配的?
__NSGlobalBlock __ 在数据区
__NSMallocBlock __ 在堆区
__NSStackBlock __ 在栈区
堆:动态分配内存,需要程序员自己申请,程序员自己管理
栈:自动分配内存,自动销毁,先入后出,栈上的内容存在自动销毁的情况
如何判断block是哪种类型?
没有访问auto变量的block是__NSGlobalBlock __ ,放在数据段
访问了auto变量的block是__NSStackBlock __
[__NSStackBlock __ copy]操作就变成了__NSMallocBlock __
对每种类型block调用copy操作后是什么结果?
__NSGlobalBlock __ 调用copy操作后,什么也不做
__NSStackBlock __ 调用copy操作后,复制效果是:从栈复制到堆;副本存储位置是堆
__NSStackBlock __ 调用copy操作后,复制效果是:引用计数增加;副本存储位置是堆
在ARC环境下,编译器会根据情况自动将栈上的block复制到堆上的几种情况?
1.block作为函数返回值时
2.将block赋值给__strong指针时
3.block作为Cocoa API中方法名含有usingBlock的方法参数时
4.block作为GCD API的方法参数时
ARC下如何解决block循环引用的问题?
三种方式:__weak、__unsafe_unretained、__block
第一种方式:__weak
Person *person = [[Person alloc] init];
// __weak Person *weakPerson = person;
__weak typeof(person) weakPerson = person;
person.block = ^{
NSLog(@"age is %d", weakPerson.age);
};
第二种方式:__unsafe_unretained
__unsafe_unretained Person *person = [[Person alloc] init];
person.block = ^{
NSLog(@"age is %d", weakPerson.age);
};
第三种方式:__block
__block Person *person = [[Person alloc] init];
person.block = ^{
NSLog(@"age is %d", person.age);
person = nil;
};
person.block();
2. 程序crash的原因?
1.赋值 setObjectForKey key不能为nil,object不能为nil [dic objectForKey:key]; key可以为nil
2. 控件属性重复拉线,一个控件拉出两个属性
3. 调用block需要判断是否为空
4. 使用NSTimer 的target被释放了,而timer还去执行定时方法,回调的时候找不到对象就会挂掉
5. 给datePicker赋值的NSDate为nil,可能因为日期格式化字符串不对
6. kvo没有移除监听,或者kvo多次注册监听
7. Range or index out of bounds
8. unrecognized selector sent to instance
9. 数组范围越界