前言
很多人都说熟悉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、内存相关(简友们提醒,不要用系统的addSubview
、removeFromSuperView
,减少干扰项)
看下面两段代码,
ViewController的代码如下
@interface ViewController () <LYButtonDelegate>
@end
@implementation ViewController
{
LYButton *testBtn;
LYButton *testBtn2;
}
- (void)viewDidLoad {
[super viewDidLoad];
testBtn = [[LYButton alloc] init];
testBtn.delegate = self;
[testBtn test];
testBtn2 = [[LYButton alloc] init];
testBtn2.delegate = self;
[testBtn2 test2];
}
- (void)onRemove:(LYButton *)btn {
if (testBtn == btn) {
testBtn = nil;
}
if (testBtn2 == btn) {
testBtn2 = nil;
}
}
LYButton的代码如下
@class LYButton;
@protocol LYButtonDelegate
- (void)onRemove:(LYButton *)btn;
@end
@implementation LYButton
- (void)test {
[self.delegate onRemove:self];
NSLog(@"%@", (self == nil) ? @"YES" : @"NO");
}
- (void)test2 {
__weak typeof (LYButton *) weakSelf = self;
dispatch_async(dispatch_get_main_queue(), ^{
[weakSelf.delegate onRemove:weakSelf];
NSLog(@"%@", (weakSelf == nil) ? @"YES" : @"NO");
NSLog(@"end");
});
}
@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中,onRemove执行之前,有-testBtn
、-test1
、self.view三个地方持有强引用,到打印log的时候两个地方的强引用;
test2中,在block中强引用了weakSelf,当block执行的时候,testBtn和test2的两个引用都已经释放,当执行完onRemove之后,最后一个引用也释放,会立刻执行dealloc
方法,weakSelf被置为nil(weak指针的用法就是在对象被回收后变成nil),故而Log输出YES;
类似,在UIButton的onClick:
回调方法中,button类的self不仅会被StackThread持有,还会被main thread dispatch持有(系统分发点击事件)。
总结
做题是一个有意思的过程,短时间的思考并得到对or错的回馈,非常适合人脑的学习模式。
希望这几道题能有所帮助。如果错误,请斧正。