前言
很多人都说熟悉UIKit,那对于常见的API是否熟悉?
多线程是前端经久不衰的考点。
大家对于Block的weak-strong dance都耳熟能详,是否清楚知道每一个引用背后的持有者,以及对象的具体释放时机?
来试试这4道精挑细选的题目。
正文
题目1、UIImage相关
看下面一段代码,
保存到相册的是什么?(从格式、形状去描述)
1
2
3
4
5
6
7
8
9
10
11
12
13
|
- (
void
)testUIImage {
UIImage *testImage;
UIGraphicsBeginImageContext(CGSizeMake(
50
,
50
));
UIView *testView = [[UIView alloc] initWithFrame:CGRectMake(
0
,
0
,
50
,
50
)];
testView.backgroundColor = [UIColor redColor];
testView.layer.cornerRadius =
25
;
testView.layer.masksToBounds = YES;
[testView.layer renderInContext:UIGraphicsGetCurrentContext()];
testImage = UIGraphicsGetImageFromCurrentImageContext();
UIGraphicsEndImageContext();
[[[ALAssetsLibrary alloc] init] writeImageToSavedPhotosAlbum:testImage.CGImage metadata:nil completionBlock:nil];
}
|
题目2、URL相关
看下面一段代码,
写下三行Log的输出,并解释下URL是什么。
1
2
3
4
5
6
7
8
9
10
11
|
- (
void
)testUrl {
NSURL *url = [NSURL URLWithString:path];
NSURL *url2 = [NSURL URLWithString:path2];
NSURL *url3 = [NSURL URLWithString:path3];
NSLog(@
"%@"
, url2);
NSLog(@
"%@"
, url3);
}
|
题目3、线程相关
看下面一段代码,
写下Log的输出,并解释为什么。
1
2
3
4
5
6
7
8
9
10
11
|
- (
void
)viewDidLoad {
[
super
viewDidLoad];
dispatch_async(dispatch_get_global_queue(
0
,
0
), ^{
NSLog(@
"before perform"
);
[self performSelector:@selector(printLog) withObject:nil afterDelay:
0
];
NSLog(@
"after perform"
);
});
}
- (
void
)printLog {
NSLog(@
"printLog"
);
}
|
题目4、内存相关
看下面两段代码,
ViewController的代码如下
1
2
3
4
5
6
7
8
9
|
- (
void
)testBtn {
LYButton *btn = [[LYButton alloc] init];
[self.view addSubview:btn];
[btn test];
[self.view addSubview:btn];
[btn test2];
}
|
LYButton的代码如下
1
2
3
4
5
6
7
8
9
10
11
12
13
14
|
@implementation LYButton
- (
void
)test {
[self removeFromSuperview];
NSLog(@
"%@"
, (self == nil) ? @
"YES"
: @
"NO"
);
}
- (
void
)test2 {
__weak
typeof
(LYButton *) weakSelf = self;
dispatch_async(dispatch_get_main_queue(), ^{
[weakSelf removeFromSuperview];
NSLog(@
"%@"
, (weakSelf == nil) ? @
"YES"
: @
"NO"
);
});
}
@end
|
写下Log的输出,并解释为什么。
答案
题目1
考察点:对常见UI操作、图片格式的了解。
内存中的testImage是非压缩的格式,保存到相册可以使用png或者jpeg格式。
-writeImageToSavedPhotosAlbum:接口默认用的jpeg的格式,如果保存png,需要将图片转成NSData,然后再保存。
testView的操作是绘制圆角按钮,然后用layer的renderInContext绘制到Context中;
结果图
题目2
考察点:对API的-URLWithString:了解,本质的知识点是URL encode。
常见的错误是在get参数添加中文,但是没有重新编码(也叫转义),导致NSURL初始化失败。
正确的做法是调用NSString的(NSString *)stringByAddingPercentEscapesUsingEncoding:(NSStringEncoding)encoding方法。
URL:Uniform Resource Locator,统一资源定位符,用的是ASCII编码。
题目3
考察点:GCD并发队列实现机制,以及performSelector的实现原理以及runloop了解。
上面这段代码,只会打印before perform和after perform,不会打印printLog。
原因:
1、GCD默认的全局并发队列,在并发执行任务的时候,会从线程池获取可执行任务的线程(如果没有就阻塞)。
2、performSelector的原理是设置一个timer到当前线程Runloop,并且是NSDefaultRunLoopMode;
3、非主线程的runloop默认是不启用;
进阶问题:加一行代码使得printLog能正常打印。
题目4
考察点:内存的引用计数。
test1中,removeFromSuperview执行之前,有-testBtn、-test1、self.view三个地方持有强引用,到打印log的时候两个地方的强引用;
test2中,在block中强引用了weakSelf,当block执行的时候,testBtn和test2的两个引用都已经释放,当执行完removeFromSuperview之后,最后一个引用也释放,会立刻执行dealloc方法,weakSelf被置为nil(weak指针的用法就是在对象被回收后变成nil),故而Log输出YES;
类似,在UIButton的onClick:回调方法中,button类的self不仅会被StackThread持有,还会被main thread dispatch持有(系统分发点击事件)。
总结
做题是一个有意思的过程,短时间的思考并得到对or错的回馈,非常适合人脑的学习模式。
希望这几道题能有所帮助。如果错误,请斧正。
原文链接:
http://www.cocoachina.com/ios/20180129/22031.html