李洪强经典面试题39-iOS 程序员 6 级考试(答案和解释)
iOS 程序员 6 级考试(答案和解释)
我是前言
1. 下面的代码分别输出什么?
@implementation Son : Father |
答案:都输出”Son”
解释:objc中super是编译器标示符,并不像self一样是一个对象,遇到向super发的方法时会转译成objc_msgSendSuper(...),而参数中的对象还是self,于是从父类开始沿继承链寻找- class这个方法,最后在NSObject中找到(若无override),此时,[self class]和[super class]已经等价了。
2. 下面的代码报错?警告?还是正常输出什么?
Father *father = [Father new]; |
答案:都输出”1”(YES)
解释:objc中:
- 不论是实例对象还是Class,都是id类型的对象(Class同样是对象)
- 实例对象的isa指向它的Class(储存所有减号方法),Class对象的isa指向元类(储存所有加号方法)
- 向一个对象(id类型)发送消息时,都是从这个对象的
isa指针指向的Class中寻找方法
回到题目,当像Father类发送一个实例方法(- responseToSelector)消息时:
- 会从它的
isa,也就是Father元类对象中寻找,由于元类中的方法都是类方法,所以自然找不到 - 于是沿继承链去父类NSObject元类中寻找,依然没有
- 由于objc对这块的设计是,NSObject的元类的父类是NSObject类(也就是我们熟悉的NSObject类),其中有所有的实例方法,因此找到了
- responseToSelector
补充:NSObject类中的所有实例方法很可能都对应实现了一个类方法(至少从开源的代码中可以看出来),如+ resonseToSelector,但并非公开的API,如果真的是这样,上面到第2步就可以找到这个方法。
再补充: 非NSObject的selector这样做无效。
3. 请求很快就执行完成,但是completionBlock很久之后才设置,还能否执行呢?
... |
答案:可以(有条件)
解释:为了方便解释,我们将其考虑成gcd的两个线性queue:main queue 和 back queue
当代码执行到sleep(100)时,这两个queue要执行的顺序看起来是这样的:
- main: *— sleep ————————-> | —setCompletionBlock—>
- back: *— network —->
于是网络请求很快回来,回调函数一般要执行如:
// 回到主线程执行回调 |
于是成了这样:
- main: *—-sleep—-> | —setCompletionBlock—> | —invoke completionBlock—->
- back: *
所以,当sleep结束后,主线程保持了调用顺序:
- main: *—setCompletionBlock—> | —invoke completionBlock—->
此时,completionBlock的执行是在setCompletionBlock,之后的,所以可以正常回调。
注:这个解释有一个有限制条件,如果用下面的方法回调,则情况就会不同了:
// 回到主线程执行回调 |
4. 不使用IB时,下面这样做有问题么?
- (void)viewDidLoad {
|
解释:不使用IB手动创建ViewController时,在viewDidLoad中并未进行位置的初始化,原来遇到过不少次这个小坑,当外部创建这个vc时:
TestViewController *vc = [[TestViewController alloc] init]; |
我们知道,ViewController的view初始化大概流程是:
- (UIView *)view {
|
所以在外部执行到vc.view.frame = CGRectMake(0, 0, 100, 100);这句话时,在赋值操作执行前,viewDidLoad就已经被调用,因而在viewDidLoad中对view frame的取值都是默认值(window的大小),而非设定值。
注: 使用IB加载时如上情况也会发生,只是一般在IB就已经有一个预设值了。
5. 下面代码输出什么?
- (void)viewDidLoad {
|
答案:输出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 级考试(答案和解释)的更多相关文章
- 李洪强经典面试题152-Runtime
李洪强经典面试题152-Runtime Runtime Runtime是什么 Runtime 又叫运行时,是一套底层的 C 语言 API,其为 iOS 内部的核心之一,我们平时编写的 OC 代码, ...
- 李洪强经典面试题145-Runloop
李洪强经典面试题145-Runloop Runloop 什么是 Runloop? 从字面上讲就是运行循环. 它内部就是do-while循环,在这个循环内部不断地处理各种任务. 一个线程对应一个Ru ...
- 李洪强经典面试题52-Block
李洪强经典面试题52-Block Block Block底层原理实现 首先我们来看四个函数 void test1() { int a = 10; void (^block)() = ^{ NSLo ...
- 李洪强经典面试题49-Objective-C
李洪强经典面试题49-Objective-C 面试笔试都是必考语法知识的.请认真复习和深入研究OC. Objective-C 方法和选择器有何不同?(Difference between method ...
- 李洪强经典面试题136-KVO-KVC
李洪强经典面试题136-KVO-KVC KVC-KVO KVC的底层实现? 当一个对象调用setValue方法时,方法内部会做以下操作: ①检查是否存在相应key的set方法,如果存在,就调用se ...
- 李洪强经典面试题53-Swift
李洪强经典面试题53-Swift Swift 网上有很多Swift的语法题,但是Swift现在语法还未稳定,所以在这里暂时不贴出语法题,可以自行搜索. Swift和Objective-C的联系 Swi ...
- 李洪强经典面试题51-KVO-KVC
李洪强经典面试题51-KVO-KVC KVC-KVO KVC的底层实现? 当一个对象调用setValue方法时,方法内部会做以下操作: ①检查是否存在相应key的set方法,如果存在,就调用set ...
- iOS 程序员 6 级考试(答案和解释)
iOS 程序员 6 级考试(答案和解释) 我是前言 1. 下面的代码分别输出什么? @implementation Son : Father- (id)init { self = [super i ...
- ios程序员6级考试(答案和解释)
http://blog.sunnyxx.com/2014/03/06/ios_exam_0_key/ 我是前言 上次发了个ios程序员6级考试题 ,还在不断补充中,开个帖子配套写答案和解释. 1. 下 ...
随机推荐
- (6)java基础知识-基本数据类型、数据类型转换
一.基本数据类型 基本的数据类型一共有四类八种 1.整型 byte: 1字节 取值范围 -128~127 short: 2字节 取值范围 -32768~32767 int: 4字节 取 ...
- 中矿大 C 石头剪刀布【决策DP*待看/codeforces原题】
时间限制:C/C++ 1秒,其他语言2秒空间限制:C/C++ 32768K,其他语言65536K64bit IO Format: %lld 题目描述 齐齐和司机正在玩剪刀石头布,不过他俩有些玩腻了,所 ...
- [Python Cookbook] Pandas Groupby
Groupby Count # Party’s Frequency of donations nyc.groupby(’Party’)[’contb receipt amt’].count() The ...
- FZU-2271 X(Floyd)
Problem 2271 X Accept: 303 Submit: 1209Time Limit: 1500 mSec Memory Limit : 32768 KB Problem ...
- C++大数板子
C++大数板子 使用样例在主函数里看就好,必要的运算符都重载了. #include <iostream> using namespace std; ;/*精度位数,自行调整*/ //1.如 ...
- 【bzoj2393】【Cirno的完美算数教室】容斥原理的剪枝应用
(上不了p站我要死了,侵权度娘背锅) 在用容斥定理时,常常会用到dfs的形式,如果枚举完所有的情况可能会超时,其剪枝的优化很是重要. Description ~Cirno发现了一种baka数,这种数呢 ...
- 面向对象-QuickHit项目
package com.ketang.game; /** * 游戏级别类 * @author * */ public class Level { private int levelNo; //各级别编 ...
- JAVA常见算法题(二)
package com.xiaowu.demo; /** * 判断101-2000之间有多少个素数,并输出所有素数. * 质数(prime number)又称素数,有无限个.质数定义为在大于1的自然数 ...
- leetcode题解:Construct Binary Tree from Preorder and Inorder Traversal (根据前序和中序遍历构造二叉树)
题目: Given preorder and inorder traversal of a tree, construct the binary tree. Note:You may assume t ...
- python解析json文件报错No JSON object could be decoded
2017-04-25 可用Nodepad++将json文件打开并以UTF8无BOM格式保存.