美文网首页
面试题不断刷新中

面试题不断刷新中

作者: 金字塔的AI | 来源:发表于2018-12-20 23:34 被阅读5次

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

@implementationSon:Father

- (id)init {

self= [superinit];

if(self) {

NSLog(@"%@",NSStringFromClass([selfclass]));

NSLog(@"%@",NSStringFromClass([superclass]));

    }

returnself;

}

@end

答案:都输出”Son”

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

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

Father *father = [Father new];

BOOLb1 = [father responseToSelector:@selector(responseToSelector:)];

BOOLb2 = [Father responseToSelector:@selector(responseToSelector:)];

NSLog(@"%d, %d", b1, b2);

答案:都输出”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很久之后才设置,还能否执行呢?

...

// 当前在主线程

[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 {

[superviewDidLoad];

CGRectframe =CGRectMake(0,0,self.view.bounds.size.width *0.5,self.bounds.size.height *0.5);

UIView*view = [[UIViewalloc] 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) {

[selfloadView];

[selfviewDidLoad];// Edit: 这句话移动到括号内,感谢@change2hao的提醒

  }

}

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

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

5. 下面代码输出什么?

- (void)viewDidLoad {

[superviewDidLoad];

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调用的结束,所以造成了循环等待,导致死锁。

后续题目继续补充中

参考:blog.sunnyxx.com

相关文章

  • 面试题不断刷新中

    1. 下面的代码分别输出什么? @implementationSon:Father - (id)init { se...

  • 不断刷新自己

    -----让生命开出五彩斑斓的花朵 2017年做了很多以前不敢去做的事情,刷新了很多领域的记录,逐渐调整...

  • 不断刷新认知

    这是我每天写一篇原创文章的第1527天 人活着,就是在不断刷新认知。 经历得多了,会有新的感悟。 比如:对于学习英...

  • 尝试不断刷新自己

    日本作家松浦弥太郎在《今天也要用心过生活》一本书中说道:我总是试图打破自己,试图反复地否定自己甚至粉碎自己,并热切...

  • 生活就是不断刷新

    “春有百花秋有月,夏有凉风冬有雪!”如此,便是人间好风景。 可是,如果春天来一场纷纷扬扬的雪花飘落,你会如何呢? ...

  • 2023不断刷新自己

    光阴似水,岁序更新。 时光的年轮悄悄刻下一圈,2023终于来了。 柏拉图说:明天对世界而言,永远是个奇迹。 新的一...

  • 历年阿里面试题汇总(2017年不断更新中)

    这是我这几天整理的关于阿里校招面试题汇总,希望对大家有所帮助。 链接:历年阿里面试题汇总(2017年不断更新中)

  • 面试题

    历年阿里面试题汇总(2017年不断更新中)面试总结Java面试重难点剖析(不断更新)Java方向如何准备BAT技术...

  • 0

    每个清晨都做 一个梦 刷新 刷新 有个数字不再攀升 新增为0 刷新 刷新 不断刷新 始终是0 窗外 己隐约有莺啼的...

  • 比特币,突然一个华丽转身,让万众叹服!

    比特币价值不断上升 刷新盘中纪录高位至6000.10美元 对于持有比特币的人来说,总有新惊喜,因为它的价值还在不断...

网友评论

      本文标题:面试题不断刷新中

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