面试题
1)@property 能使用哪些关键字、作用是什么;下面这4种写法的区别?
@property (nonatomic, strong) NSArray *array0;//mutableArray copy 成array0时,如果array0种的值被改了,会出什么问题呢
@property (nonatomic, copy) NSArray *array1;//常规写法
@property (nonatomic, strong) NSMutableArray *array2;//标准写法
@property (nonatomic, copy) NSMutableArray *array3;// 把Array mutablecopy 成 array3后,array3其实还是 Array对象,如果向其中插入对象的话,则会造成crash
copy的深浅拷贝只说是不正确的。
copy 是相对于mutablecopy来说,Array mutablecopy可以变成MutableArry, MutableArry可以copy成Array。同理还有NSdiction。
一个Array mutablecopy 成一个 MutableArray后,Array里面的东西还是原来的东西,里面的元素并没有重新new一个新的东西申请新的内存,这个是浅拷贝,并不是之前说的深拷贝。
2) 下边对象分别都在什么时候释放
- (void)ARCProblem{
id obj0 = @"sunnyxx"; //会一直在
__weak id obj1 = obj0; //__weak obj1的地址指向obj0的对象,所以只要obj0不释放obj1就不会被释放
__strong id obj2 = [NSObject new];//obj2 会在用完的时候,obj2被释放
__weak id obj3 = [NSObject new]; //obj3 在赋值之后就会被释放,因为没有人强引用obj3,所以在这行结束 obj3就会被释放
{
id obj4 = [NSObject new];// 在{}区域结束的时候会被释放
}
__autoreleasing id obj5 = [NSObject new];// obj5是个自动释放的对象,obj5会被加到自动释放池中,会在一个最近的自动释放池中被释放,如果是ARC外面有个autoreleasPool 则会在那个时候被释放,如果是在viewdidload中,则会在runloop的autoreleasPool中释放。
__unsafe_unretained id obj6 = self;//相当于retain,obj6不会对self进行影响,理论上obj6在任何时候都会被释放。 unsafe不保证对象时刻都有,可能随时销毁。
}
3) 会发生什么
- (void)viewDidLoad {
[super viewDidLoad];
self.view = nil;
//view Did Load 是在view被load的时候被调用,self.view 是个懒加载的属性,第一次在.view的时候,会load一下view相关方法,在加载完成后悔通知VC执行viewDidLoad方法
// 所以在self.view = nil 的时候
// 很有可能出现递归,但是如果没有再次调用 self.view 也可能不会出现递归,出现黑屏现象
}
4) block 循环引用的三种解决方案
//如何解决下面代码的问题
- (void)blockRetainCycleProblem{
self.block = ^{
NSLog(@"%@",@[self]);
};
}
// ===== 三种解决办法
- (void)blockRetainCycleProbleAnswer0{
__weak typeof(self) weakSelf;
self.block = ^{
NSLog(@"%@",@[weakSelf]);
};
}
- (void)blockRetainCycleProbleAnswer1{
__weak typeof(self) weakSelf;
self.block = ^{
__weak typeof(weakSelf) strongSelf = weakSelf;
NSLog(@"%@", @[strongSelf]);
// 确保在 block块内 self不会被释放, 如果放倒数组里带出block区域,则可能造成crash,self可能被释放掉
};
}
- (void)blockRetainCycleProbleAnswer2{
__weak typeof(self) weakSelf;
self.block = ^{
__weak typeof(weakSelf) strongSelf = weakSelf;
if (strongSelf) {
NSLog(@"%@",@[strongSelf]);
}
};
}
5)Block内存管理Extension
///下面的 self 用不用weak
- (void)blockRetainCycleProblemExt{
// 下面的 self 用不用weak
[UIView animateWithDuration:1 animations:^{
self.view.frame = CGRectMake(1, 2, 3, 4);
// 这行block 是瞬间执行的,在这行代码走完,会立即调用这个block方法
}];
//那这个呢
[UIView animateWithDuration:1 delay:1000 options:0 animations:^{
self.view.frame = CGRectMake(4, 1, 2, 3);
// 这个也是瞬间执行
} completion:nil];
}
6)代码规范
#pragma mark - 代码规范 (问题)
typedef enum {
UserSex_Man,
UserSex_Womain
}UserSex;
@interface UserModel : NSObject
@property(nonatomic, strong) NSString *name;
@property(assign, nonatomic) int age;
@property (nonatomic, assign) UserSex sex;
-(id)initUserModelWithUserName: (NSString*)name withAge:(int)age;
-(void)doLogIn;
@end
#pragma mark - 正确的代码规范
typedef NS_ENUM(NSUInteger, XXUserGender) {
XXUserGenderUndefined,
XXUserGenderMale,
XXUserGenderFemale
};
@interface XXUserModel : NSObject
@property (nonatomic, strong) NSString *name;
@property (nonatomic, assign) int age;
@property (nonatomic, assign) XXUserGender gender;
-(instancetype)initWithUsername:(NSString *)name age:(NSInteger)age;
-(void)login;
@end