iOS 程序员 6 级考试(答案和解释)

 

我是前言


1. 下面的代码分别输出什么?

@implementation Son : Father
- (id)init {
self = [super init];
if (self) {
NSLog(@"%@", NSStringFromClass([self class]));
NSLog(@"%@", NSStringFromClass([super class]));
}
return self;
}
@end

答案:都输出”Son”
解释:objc中super是编译器标示符,并不像self一样是一个对象,遇到向super发的方法时会转译成objc_msgSendSuper(...),而参数中的对象还是self,于是从父类开始沿继承链寻找- class这个方法,最后在NSObject中找到(若无override),此时,[self class]和[super class]已经等价了。

2. 下面的代码报错?警告?还是正常输出什么?

Father *father = [Father new];
BOOL b1 = [father responseToSelector:@selector(responseToSelector:)];
BOOL b2 = [Father responseToSelector:@selector(responseToSelector:)];
NSLog(@"%d, %d", b1, b2);

答案:都输出”1”(YES)
解释:objc中:

  • 不论是实例对象还是Class,都是id类型的对象(Class同样是对象)
  • 实例对象的isa指向它的Class(储存所有减号方法),Class对象的isa指向元类(储存所有加号方法)
  • 向一个对象(id类型)发送消息时,都是从这个对象的isa指针指向的Class中寻找方法

回到题目,当像Father类发送一个实例方法(- responseToSelector)消息时:

  1. 会从它的isa,也就是Father元类对象中寻找,由于元类中的方法都是类方法,所以自然找不到
  2. 于是沿继承链去父类NSObject元类中寻找,依然没有
  3. 由于objc对这块的设计是,NSObject的元类的父类是NSObject类(也就是我们熟悉的NSObject类),其中有所有的实例方法,因此找到了- responseToSelector

补充:NSObject类中的所有实例方法很可能都对应实现了一个类方法(至少从开源的代码中可以看出来),如+ resonseToSelector,但并非公开的API,如果真的是这样,上面到第2步就可以找到这个方法。
再补充: 非NSObject的selector这样做无效。

3. 请求很快就执行完成,但是completionBlock很久之后才设置,还能否执行呢?

...
// 当前在主线程
[request startAsync]; // 后台线程异步调用,完成后会在主线程调用completionBlock
sleep(100); // sleep主线程,使得下面的代码在后台线程完成后才能执行
[request setCompletionBlock:^{
NSLog(@"Can I be printed?");
}];
...

答案:可以(有条件)
解释:为了方便解释,我们将其考虑成gcd的两个线性queue:main queue 和 back queue

当代码执行到sleep(100)时,这两个queue要执行的顺序看起来是这样的:

  • main: *— sleep ————————-> | —setCompletionBlock—>
  • back: *— network —->

于是网络请求很快回来,回调函数一般要执行如:

// 回到主线程执行回调
dispatch_async(dispatch_get_main_queue(), ^{
if (self.completionBlock) self.completionBlock();
});

于是成了这样:

  • main: *—-sleep—-> | —setCompletionBlock—> | —invoke completionBlock—->
  • back: *

所以,当sleep结束后,主线程保持了调用顺序:

  • main: *—setCompletionBlock—> | —invoke completionBlock—->

此时,completionBlock的执行是在setCompletionBlock,之后的,所以可以正常回调。

注:这个解释有一个有限制条件,如果用下面的方法回调,则情况就会不同了:

// 回到主线程执行回调
if (self.completionBlock) {
dispatch_async(dispatch_get_main_queue(), ^{
self.completionBlock();
});

4. 不使用IB时,下面这样做有问题么?

- (void)viewDidLoad {
[super viewDidLoad];
CGRect frame = CGRectMake(0, 0, self.view.bounds.size.width * 0.5, self.bounds.size.height * 0.5);
UIView *view = [[UIView alloc] initWithFrame:frame];
[self.view addSubview:view];
}

解释:不使用IB手动创建ViewController时,在viewDidLoad中并未进行位置的初始化,原来遇到过不少次这个小坑,当外部创建这个vc时:

TestViewController *vc = [[TestViewController alloc] init];
vc.view.frame = CGRectMake(0, 0, 100, 100);
//...

我们知道,ViewController的view初始化大概流程是:

- (UIView *)view {
if (!_view) {
[self loadView];
[self viewDidLoad]; // Edit: 这句话移动到括号内,感谢@change2hao的提醒
}
}

所以在外部执行到vc.view.frame = CGRectMake(0, 0, 100, 100);这句话时,在赋值操作执行前,viewDidLoad就已经被调用,因而在viewDidLoad中对view frame的取值都是默认值(window的大小),而非设定值。

注: 使用IB加载时如上情况也会发生,只是一般在IB就已经有一个预设值了。

5. 下面代码输出什么?

- (void)viewDidLoad {
[super viewDidLoad]; NSLog(@"1");
dispatch_sync(dispatch_get_main_queue(), ^{
NSLog(@"2");
});
NSLog(@"3");
}

答案:输出1之后程序死锁
解释:dispatch_sync文档中提到:

Calls to dispatch_sync() targeting the current queue will result in dead-lock. Use of dispatch_sync() is also subject to the same multi-party dead-lock problems that may result from the use of a mutex. Use of dispatch_async() is preferred.

sync到当前线程的block将会引起死锁,所以只会Log出1来后主线程就进入死锁状态,不会继续执行。
究其原因,还要看dispatch_sync做的事,它将一个block插入到queue中,这点和async没有区别,区别在于sync会等待到这个block执行完成后才回到调用点继续执行,而这个block的执行还依仗着viewDidLoad中dispatch_sync调用的结束,所以造成了循环等待,导致死锁。


后续题目继续补充中

李洪强经典面试题39-iOS 程序员 6 级考试(答案和解释)的更多相关文章

  1. 李洪强经典面试题152-Runtime

    李洪强经典面试题152-Runtime   Runtime Runtime是什么 Runtime 又叫运行时,是一套底层的 C 语言 API,其为 iOS 内部的核心之一,我们平时编写的 OC 代码, ...

  2. 李洪强经典面试题145-Runloop

    李洪强经典面试题145-Runloop   Runloop 什么是 Runloop? 从字面上讲就是运行循环. 它内部就是do-while循环,在这个循环内部不断地处理各种任务. 一个线程对应一个Ru ...

  3. 李洪强经典面试题52-Block

    李洪强经典面试题52-Block   Block Block底层原理实现 首先我们来看四个函数 void test1() { int a = 10; void (^block)() = ^{ NSLo ...

  4. 李洪强经典面试题49-Objective-C

    李洪强经典面试题49-Objective-C 面试笔试都是必考语法知识的.请认真复习和深入研究OC. Objective-C 方法和选择器有何不同?(Difference between method ...

  5. 李洪强经典面试题136-KVO-KVC

    李洪强经典面试题136-KVO-KVC   KVC-KVO KVC的底层实现? 当一个对象调用setValue方法时,方法内部会做以下操作: ①检查是否存在相应key的set方法,如果存在,就调用se ...

  6. 李洪强经典面试题53-Swift

    李洪强经典面试题53-Swift Swift 网上有很多Swift的语法题,但是Swift现在语法还未稳定,所以在这里暂时不贴出语法题,可以自行搜索. Swift和Objective-C的联系 Swi ...

  7. 李洪强经典面试题51-KVO-KVC

    李洪强经典面试题51-KVO-KVC   KVC-KVO KVC的底层实现? 当一个对象调用setValue方法时,方法内部会做以下操作: ①检查是否存在相应key的set方法,如果存在,就调用set ...

  8. iOS 程序员 6 级考试(答案和解释)

    iOS 程序员 6 级考试(答案和解释)   我是前言 1. 下面的代码分别输出什么? @implementation Son : Father- (id)init { self = [super i ...

  9. ios程序员6级考试(答案和解释)

    http://blog.sunnyxx.com/2014/03/06/ios_exam_0_key/ 我是前言 上次发了个ios程序员6级考试题 ,还在不断补充中,开个帖子配套写答案和解释. 1. 下 ...

随机推荐

  1. (15)C#集合

    http://blog.csdn.net/hcw_peter/article/details/3980723 集合分为非泛型集合和泛型集合 ,泛型集合可以指定放入集合中的类型. 一.非泛性集合 引用命 ...

  2. 整数划分问题(记忆化搜索和DP方法)

    一. 问题 现在有一正整数N,要把它分为若干正整数之和,问有多少种本质不同的分法? (1)其中最大数不超过m, 有多少种分法? (2)分割后的正整数的数目不超过m个, 有多少种分法? (3)分成最大数 ...

  3. Python实现微信小程序支付功能

    由于最近自己在做小程序的支付,就在这里简单介绍一下讲一下用python做小程序支付这个流程.当然在进行开发之前还是建议读一下具体的流程,清楚支付的过程. 1.支付交互流程 当然具体的参数配置可以参考官 ...

  4. HDU6383 2018 “百度之星”程序设计大赛 - 初赛(B) 1004-p1m2 (二分)

    原题地址 p1m2 Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 131072/131072 K (Java/Others)Total ...

  5. 洛谷——2639[USACO09OCT]Bessie的体重问题Bessie's We…——01

    题目描述 Bessie像她的诸多姊妹一样,因为从Farmer John的草地吃了太多美味的草而长出了太多的赘肉.所以FJ将她置于一个及其严格的节食计划之中.她每天不能吃多过H (5 <= H & ...

  6. SPOJ 10628 COT - Count on a tree(在树上建立主席树)(LCA)

    COT - Count on a tree #tree You are given a tree with N nodes.The tree nodes are numbered from 1 to ...

  7. Listener监听器笔记1

    1.常用的Web事件监听器接口: 1.ServletContextListener:用于监听Web应用的启动和关闭. 2.ServletContextAttributeListener:用于监听Ser ...

  8. Linux CURL的安装

    Linux CURL的安装  Linux CURL的安装   --获得安装包,从网上直接下载或者其他途径,这里直接wget# wget http://curl.haxx.se/download/cur ...

  9. Ubuntu 16.04下用Wine运行的软件出现方块的解决思路(应该是兼容现在所有平台的Wine碰到这个的问题)

    说明: 1.我使用的是深度的deepin-wine,版本为1.9.0,参考:http://www.cnblogs.com/EasonJim/p/8016674.html 2.这种问题没有一定的解决的方 ...

  10. SQL 存储过程入门(事务)

    本篇我们来讲一下事务处理技术. 为什么要使用事务呢,事务有什么用呢,举个例子. 假设我们现在有个业务,当做成功某件事情的时候要向2张表中插入数据,A表,B表,我们插入的顺序是先插入A,再插入B表,如果 ...