iOS面试题----实践向

作者: 落影loyinglin | 来源:发表于2018-01-29 08:46 被阅读1961次

    前言

    很多人都说熟悉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错的回馈,非常适合人脑的学习模式。
    希望这几道题能有所帮助。如果错误,请斧正。

    相关文章

      网友评论

      • 苹果API搬运工:题目4的解释应该有点问题,改一下test2方法就能看出来:
        - (void)test2 {
        __weak typeof (LYButton *) weakSelf = self;
        [weakSelf.delegate onRemove:weakSelf];
        NSLog(@"%@", (weakSelf == nil) ? @"YES" : @"NO");//输出YES
        NSLog(@"end");//输出end
        NSLog(@"%@", (self == nil) ? @"YES" : @"NO");//输出NO
        dispatch_async(dispatch_get_main_queue(), ^{
        });

        }
        去掉那个block,打印weakSelf是YES,打印self就是NO;
      • Sunrain16:题目3
        [[NSRunLoop currentRunLoop] run];
        是这个样子吧。按照作者您的意思是performSelector会设置一个timer在当前线程的runloop中。但是这个线程又不是主线程,所以才会不执行。解决的方法:
        1.performSelector的方法在主线程运行
        2.重新启动当前的线程的runloop
        但是两种解决方法执行的顺序是不一样的。
        落影loyinglin:@Sunrain16 只加这一行,看你加的位置了。
      • menttofly:记得腾讯音乐的面试官出的一道题,NSData输出的16进制结果转回NSData:joy:
        落影loyinglin:@menttofly 我只是好奇哪个面试官,你知道英文名吗
        menttofly:@落影loyinglin 是用<6c756f79 696e67>作为输入字符串,转回原来对应的data...对大牛你来说很简单哈哈哈
        落影loyinglin:@menttofly 有点容易?
      • 不上火喝纯净水:这道题是假定removeFromSuperview执行后view的引用计数只会-1的前提下成立的。
        然而removeFromSuperview方法并不是这么回事,按照我的测试结果removeFromSuperview对view引用计数的影响如下:
        1.release 1次 = 引用-1,这里-1和addSubview方法的+1对应;
        2.retain 并将view添加到当前自动释放池中 2次 = 引用计数+2
        测试结果说明了removeFromSuperview方法执行以后还会执行其他的方法,view还需要参与其他的操作,view此时不能直接释放,所以retain并放到自动释放池中。等事件循环结束引用计数-2.

        总的来说这道题作为考察内存释放原理是不合适的,因为像removeFromSuperview这样的系统方法指不定苹果在下个版本又会发生变化,view的引用计数可能还会不一样。
        落影loyinglin:多谢提醒,已经换用新的描述方式。其实这个题的原本思路就是vc变量持有,而不是addSubView,选用这个api是想减少描述代码,方便理解,没想到走了弯路。(顺带说下,之前那段代码,我的Xcode输出是YES)
        啊哈呵:有道理,有可能removeFromSuperview里面搞鬼
      • SomeBoy:题目4 , run 了一下, test2 输出是NO, block 貌似会对self 造成引用吧
        小Ping平:@SomeBoy 你们的结果怎么样呢?block中使用weak只是防止相互持有,应该是会持有的。
        SomeBoy:@落影loyinglin
        button test2 方法代码
        - (void)test2 {

        NSLog(@"[In Test2] retain count = %ld\n",CFGetRetainCount((__bridge CFTypeRef)(self)));
        __weak typeof (LYButton *) weakSelf = self;
        dispatch_async(dispatch_get_main_queue(), ^{
        NSLog(@"[In Block][Before remove]retain count = %ld\n",CFGetRetainCount((__bridge CFTypeRef)(weakSelf)));
        [weakSelf removeFromSuperview];
        NSLog(@"[In Block][After remove]retain count = %ld\n",CFGetRetainCount((__bridge CFTypeRef)(weakSelf)));
        NSLog(@"%@", (weakSelf == nil) ? @"YES" : @"NO");
        });
        }
        ======log 输出
        2018-01-29 15:17:28.482401+0800 Demo3[34789:1065078] [In Test2] retain count = 2
        2018-01-29 15:17:28.488815+0800 Demo3[34789:1065078] [In Block][Before remove]retain count = 3
        2018-01-29 15:17:28.489088+0800 Demo3[34789:1065078] [In Block][After remove]retain count = 3
        2018-01-29 15:17:28.489283+0800 Demo3[34789:1065078] NO
        落影loyinglin:@SomeBoy 你看看哪里持有了这个引用

      本文标题:iOS面试题----实践向

      本文链接:https://www.haomeiwen.com/subject/wqvloxtx.html