前言
很多人都说熟悉UIKit,那对于常见的API是否熟悉?
多线程是前端经久不衰的考点。
大家对于Block的weak-strong dance都耳熟能详,是否清楚知道每一个引用背后的持有者,以及对象的具体释放时机?
来试试这4道精挑细选的题目。
正文
题目1、UIImage相关
看下面一段代码,
保存到相册的是什么?(从格式、形状去描述)
- (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是什么。
- (void)testUrl {
NSString *path = @"https://www.baidu.com/";
NSString *path2 = @"http://fanyi.baidu.com/translate?query=#auto/zh/";
NSString *path3 = @"http://fanyi.baidu.com/translate?query=#zh/en/测试";
NSURL *url = [NSURL URLWithString:path];
NSURL *url2 = [NSURL URLWithString:path2];
NSURL *url3 = [NSURL URLWithString:path3];
NSLog(@"%@", url);
NSLog(@"%@", url2);
NSLog(@"%@", url3);
}题目3、线程相关
看下面一段代码,
写下Log的输出,并解释为什么。
- (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的代码如下
- (void)testBtn {
LYButton *btn = [[LYButton alloc] init];
[self.view addSubview:btn];
[btn test];
[self.view addSubview:btn];
[btn test2];
}LYButton的代码如下
@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错的回馈,非常适合人脑的学习模式。
希望这几道题能有所帮助。如果错误,请斧正。
作者:落影loyinglin
链接:https://www.jianshu.com/p/941039aba684