前言
很多人都说熟悉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