iOS面试题iOS开发专题计算机杂谈

iOS面试题----实践向

2018-01-29  本文已影响1961人  落影loyinglin

前言

很多人都说熟悉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、内存相关(简友们提醒,不要用系统的addSubviewremoveFromSuperView,减少干扰项)

看下面两段代码,
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 performafter 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错的回馈,非常适合人脑的学习模式。
希望这几道题能有所帮助。如果错误,请斧正。

上一篇下一篇

猜你喜欢

热点阅读