iOS超全面试题,面试前看一看,不错<上篇>

以前看了不少面试题,感受要不是不够就是过于冗余,因而我将网上的一些面试题进行了删减和重排,如今分享给你们。(题目来源于网络,侵删)
java

1. Object-c的类能够多重继承么?能够实现多个接口么?Category是什么?重写一个类的方式用继承好仍是分类好?为何?ios

答: Object-c的类不能够多重继承;能够实现多个接口,经过实现多个接口能够完成C++的多重继承;Category是类别,通常状况用分类好,用Category去重写类的方法,仅对本Category有效,不会影响到其余类与原有类的关系。c++

2. #import 跟#include 又什么区别,@class呢, #import<> 跟 #import””又什么区别?程序员

答:#import是Objective-C导入头文件的关键字,#include是C/C++导入头文件的关键字,使用#import头文件会自动只导入一次,不会重复导入,至关于#include和#pragma once;@class告诉编译器某个类的声明,当执行时,才去查看类的实现文件,能够解决头文件的相互包含;#import<>用来包含系统的头文件,#import””用来包含用户头文件。web

3. 属性readwrite,readonly,assign,retain,copy,nonatomic 各是什么做用,在那种状况下用?面试

答:objective-c

1). readwrite 是可读可写特性;须要生成getter方法和setter方法时数据库

2). readonly 是只读特性 只会生成getter方法 不会生成setter方法 ;不但愿属性在类外改变编程

3). assign 是赋值特性,setter方法将传入参数赋值给实例变量;仅设置变量时;设计模式

4). retain 表示持有特性,setter方法将传入参数先保留,再赋值,传入参数的retaincount会+1;

5). copy 表示赋值特性,setter方法将传入对象复制一份;须要彻底一份新的变量时。

6).nonatomic 非原子操做,决定编译器生成的setter getter是不是原子操做,atomic表示多线程安全,通常使用nonatomic

4.写一个setter方法用于完成@property (nonatomic,retain)NSString *name,写一个setter方法用于完成@property(nonatomic,copy)NSString *name

答:

1

2

3

4

5

6

7

8

9

10

11

12- (void) setName:(NSString*) str

{

[str retain];

[name release];

name = str;

}

- (void)setName:(NSString *)str

{

id t = [str copy];

[name release];

name = t;

}

5.对于语句NSString*obj = [[NSData alloc] init]; obj在编译时和运行时分别时什么类型的对象?

答: 编译时是NSString的类型;运行时是NSData类型的对象

6.常见的object-c的数据类型有那些, 和C的基本数据类型有什么区别?如:NSInteger和int

答:object-c的数据类型有NSString,NSNumber,NSArray,NSMutableArray,NSData等等,这些都是class,建立后即是对象,而C语言的基本数据类型int,只是必定字节的内存空间,用于存放数值;NSInteger是基本数据类型,并非NSNumber的子类,固然也不是NSObject的子类。NSInteger是基本数据类型Int或者Long的别名(NSInteger的定义typedef long NSInteger),它的区别在于,NSInteger会根据系统是32位仍是64位来决定是自己是int仍是Long。

7.id 声明的对象有什么特性?

答:Id 声明的对象具备运行时的特性,便可以指向任意类型的objcetive-c的对象;

8.Objective-C如何对内存管理的,说说你的见解和解决方法?

答:Objective-C的内存管理主要有三种方式ARC(自动内存计数)、手动内存计数、内存池。

1). (Garbage Collection)自动内存计数:这种方式和java相似,在你的程序的执行过程当中。始终有一个高人在背后准确地帮你收拾垃圾,你不用考虑它何时开始工做,怎样工做。你只须要明白,我申请了一段内存空间,当我再也不使用从而这段内存成为垃圾的时候,我就完全的把它忘记掉,反正那个高人会帮我收拾垃圾。遗憾的是,那个高人须要消耗必定的资源,在携带设备里面,资源是紧俏商品因此iPhone不支持这个功能。因此“Garbage Collection”不是本入门指南的范围,对“Garbage Collection”内部机制感兴趣的同窗能够参考一些其余的资料,不过说老实话“Garbage Collection”不大适合适初学者研究。

解决: 经过alloc – initial方式建立的, 建立后引用计数+1, 此后每retain一次引用计数+1, 那么在程序中作相应次数的release就行了.

2). (Reference Counted)手动内存计数:就是说,从一段内存被申请以后,就存在一个变量用于保存这段内存被使用的次数,咱们暂时把它称为计数器,当计数器变为0的时候,那么就是释放这段内存的时候。好比说,当在程序A里面一段内存被成功申请完成以后,那么这个计数器就从0变成1(咱们把这个过程叫作alloc),而后程序B也须要使用这个内存,那么计数器就从1变成了2(咱们把这个过程叫作retain)。紧接着程序A再也不须要这段内存了,那么程序A就把这个计数器减1(咱们把这个过程叫作release);程序B也再也不须要这段内存的时候,那么也把计数器减1(这个过程仍是release)。当系统(也就是Foundation)发现这个计数器变 成员了0,那么就会调用内存回收程序把这段内存回收(咱们把这个过程叫作dealloc)。顺便提一句,若是没有Foundation,那么维护计数器,释放内存等等工做须要你手工来完成。

解决:通常是由类的静态方法建立的, 函数名中不会出现alloc或init字样, 如[NSString string]和[NSArray arrayWithObject:], 建立后引用计数+0, 在函数出栈后释放, 即至关于一个栈上的局部变量. 固然也能够经过retain延长对象的生存期.

3). (NSAutoRealeasePool)内存池:能够经过建立和释放内存池控制内存申请和回收的时机.

解决:是由autorelease加入系统内存池, 内存池是能够嵌套的, 每一个内存池都须要有一个建立释放对, 就像main函数中写的同样. 使用也很简单, 好比[[[NSString alloc]initialWithFormat:@”Hey you!”] autorelease], 即将一个NSString对象加入到最内层的系统内存池, 当咱们释放这个内存池时, 其中的对象都会被释放.

9. 原子(atomic)跟非原子(non-atomic)属性有什么区别?

答:

1). atomic提供多线程安全。是防止在写未完成的时候被另一个线程读取,形成数据错误

2). non-atomic:在本身管理内存的环境中,解析的访问器保留并自动释放返回的值,若是指定了 nonatomic ,那么访问器只是简单地返回这个值。

10. 看下面的程序,第一个NSLog会输出什么?这时str的retainCount是多少?第二个和第三个呢? 为何?

1

2

3

4

5

6

7

8

9

10

11NSMutableArray* ary = [[NSMutableArray array] retain];

NSString *str = [NSString stringWithFormat:@"test"];

[str retain];

[aryaddObject:str];

NSLog(@”%@%d”,str,[str retainCount]);

[str retain];

[str release];

[str release];

NSLog(@”%@%d”,str,[str retainCount]);

[aryremoveAllObjects];

NSLog(@”%@%d”,str,[str retainCount]);

str的retainCount建立+1,retain+1,加入数组自动+1 3

retain+1,release-1,release-1 2

数组删除全部对象,全部数组内的对象自动-1 1

11. 内存管理的几条原则时什么?按照默认法则.那些关键字生成的对象须要手动释放?在和property结合的时候怎样有效的避免内存泄露?

答:谁申请,谁释放

遵循Cocoa Touch的使用原则;

内存管理主要要避免“过早释放”和“内存泄漏”,对于“过早释放”须要注意@property设置特性时,必定要用对特性关键字,对于“内存泄漏”,必定要申请了要负责释放,要细心。

关键字alloc 或new 生成的对象须要手动释放;

设置正确的property属性,对于retain须要在合适的地方释放,

12.如何对iOS设备进行性能测试?

答: Profile-> Instruments ->Time Profiler

13. Object C中建立线程的方法是什么?若是在主线程中执行代码,方法是什么?若是想延时执行代码、方法又是什么?

答:线程建立有三种方法:使用NSThread建立、使用GCD的dispatch、使用子类化的NSOperation,而后将其加入NSOperationQueue;在主线程执行代码,方法是performSelectorOnMainThread,若是想延时执行代码能够用performSelector:onThread:withObject:waitUntilDone:

14. MVC设计模式是什么? 你还熟悉什么设计模式?

答:

设计模式:并非一种新技术,而是一种编码经验,使用好比java中的接口,iphone中的协议,继承关系等基本手段,用比较成熟的逻辑去处理某一种类型的事情,总结为所谓设计模式。面向对象编程中,java已经概括了23种设计模式。

mvc设计模式 :模型,视图,控制器,能够将整个应用程序在思想上分红三大块,对应是的数据的存储或处理,前台的显示,业务逻辑的控制。 Iphone自己的设计思想就是遵循mvc设计模式。其不属于23种设计模式范畴。

代理模式:代理模式给某一个对象提供一个代理对象,并由代理对象控制对源对象的引用.好比一个工厂生产了产品,并不想直接卖给用户,而是搞了不少代理商,用户能够直接找代理商买东西,代理商从工厂进货.常见的如QQ的自动回复就属于代理拦截,代理模式在iphone中获得普遍应用.

单例模式:说白了就是一个类不经过alloc方式建立对象,而是用一个静态方法返回这个类的对象。系统只须要拥有一个的全局对象,这样有利于咱们协调系统总体的行为,好比想得到[UIApplication sharedApplication];任何地方调用均可以获得 UIApplication的对象,这个对象是全局惟一的。

观察者模式: 当一个物体发生变化时,会通知全部观察这个物体的观察者让其作出反应。实现起来无非就是把全部观察者的对象给这个物体,当这个物体的发生改变,就会调用遍历全部观察者的对象调用观察者的方法从而达到通知观察者的目的。

工厂模式:

1

2

3

4

5

6

7

8publicclassFactory{

publicstaticSample creator(int which){

if(which==1)

returnnewSampleA();

elseif(which==2)

returnnewSampleB();

}

}

15 浅复制和深复制的区别?

答:浅层复制:只复制指向对象的指针,而不复制引用对象自己。

深层复制:复制引用对象自己。

意思就是说我有个A对象,复制一份后获得A_copy对象后,对于浅复制来讲,A和A_copy指向的是同一个内存资源,复制的只不过是是一个指针,对象自己资源

仍是只有一份,那若是咱们对A_copy执行了修改操做,那么发现A引用的对象一样被修改,这其实违背了咱们复制拷贝的一个思想。深复制就好理解了,内存中存在了

两份独立对象自己。

用网上一哥们通俗的话将就是:

浅复制比如你和你的影子,你完蛋,你的影子也完蛋

深复制比如你和你的克隆人,你完蛋,你的克隆人还活着。

16. 类别的做用?继承和类别在实现中有何区别?

答:category 能够在不获悉,不改变原来代码的状况下往里面添加新的方法,只能添加,不能删除修改,而且若是类别和原来类中的方法产生名称冲突,则类别将覆盖原来的方法,由于类别具备更高的优先级。

类别主要有3个做用:

1).将类的实现分散到多个不一样文件或多个不一样框架中。

2).建立对私有方法的前向引用。

3).向对象添加非正式协议。

继承能够增长,修改或者删除方法,而且能够增长属性。

17. 类别和类扩展的区别。

答:category和extensions的不一样在于 后者能够添加属性。另外后者添加的方法是必需要实现的。

extensions能够认为是一个私有的Category。

18. oc中的协议和java中的接口概念有何不一样?

答:OC中的代理有2层含义,官方定义为 formal和informal protocol。前者和Java接口同样。

informal protocol中的方法属于设计模式考虑范畴,不是必须实现的,可是若是有实现,就会改变类的属性。

其实关于正式协议,类别和非正式协议我很早前学习的时候大体看过,也写在了学习教程里

“非正式协议概念其实就是类别的另外一种表达方式“这里有一些你可能但愿实现的方法,你可使用他们更好的完成工做”。

这个意思是,这些是可选的。好比我门要一个更好的方法,咱们就会申明一个这样的类别去实现。而后你在后期能够直接使用这些更好的方法。

这么看,总以为类别这玩意儿有点像协议的可选协议。”

如今来看,其实protocal已经开始对二者都统一和规范起来操做,由于资料中说“非正式协议使用interface修饰“,

如今咱们看到协议中两个修饰词:“必须实现(@requied)”和“可选实现(@optional)”。

19. 什么是KVO和KVC?

答:KVC:键 – 值编码是一种间接访问对象的属性使用字符串来标识属性,而不是经过调用存取方法,直接或经过实例变量访问的机制。

不少状况下能够简化程序代码。apple文档其实给了一个很好的例子。

KVO:键值观察机制,他提供了观察某一属性变化的方法,极大的简化了代码。

具体用看到嗯哼用到过的一个地方是对于按钮点击变化状态的的监控。

好比我自定义的一个button

1

2

3

4

5

6

7

8[self addObserver:self forKeyPath:@"highlighted"options:0 context:nil];

#pragma mark KVO

- (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary *)change context:(void *)context

{

if([keyPath isEqualToString:@"highlighted"] ) {

[self setNeedsDisplay];

}

}

对于系统是根据keypath去取的到相应的值发生改变,理论上来讲是和kvc机制的道理是同样的。

对于kvc机制如何经过key寻找到value:

“当经过KVC调用对象时,好比:[self valueForKey:@”someKey”]时,程序会自动试图经过几种不一样的方式解析这个调用。首先查找对象是否带有 someKey 这个方法,若是没找到,会继续查找对象是否带有someKey这个实例变量(iVar),若是尚未找到,程序会继续试图调用 -(id) valueForUndefinedKey:这个方法。若是这个方法仍是没有被实现的话,程序会抛出一个NSUndefinedKeyException异常错误。

(cocoachina.com注:Key-Value Coding查找方法的时候,不只仅会查找someKey这个方法,还会查找getsomeKey这个方法,前面加一个get,或者_someKey以及_getsomeKey这几种形式。同时,查找实例变量的时候也会不只仅查找someKey这个变量,也会查找_someKey这个变量是否存在。)

设计valueForUndefinedKey:方法的主要目的是当你使用-(id)valueForKey方法从对象中请求值时,对象可以在错误发生前,有最后的机会响应这个请求。这样作有不少好处,下面的两个例子说明了这样作的好处。“

来至cocoa,这个说法应该挺有道理。

由于咱们知道button倒是存在一个highlighted实例变量.所以为什么上面咱们只是add一个相关的keypath就好了,

能够按照kvc查找的逻辑理解,就说的过去了。

20. 代理的做用?

答:代理的目的是改变或传递控制链。容许一个类在某些特定时刻通知到其余类,而不须要获取到那些类的指针。能够减小框架复杂度。

另一点,代理能够理解为java中的回调监听机制的一种相似。

21. oc中可修改和不能够修改类型。

答:可修改不可修改的集合类。这个我我的简单理解就是可动态添加修改和不可动态添加修改同样。

好比NSArray和NSMutableArray。前者在初始化后的内存控件就是固定不可变的,后者能够添加等,能够动态申请新的内存空间。

22. 咱们说的oc是动态运行时语言是什么意思?

答:多态。 主要是将数据类型的肯定由编译时,推迟到了运行时。

这个问题其实浅涉及到两个概念,运行时和多态。

简单来讲,运行时机制使咱们直到运行时才去决定一个对象的类别,以及调用该类别对象指定方法。

多态:不一样对象以本身的方式响应相同的消息的能力叫作多态。意思就是假设生物类(life)都用有一个相同的方法-eat;

那人类属于生物,猪也属于生物,都继承了life后,实现各自的eat,可是调用是咱们只需调用各自的eat方法。

也就是不一样的对象以本身的方式响应了相同的消息(响应了eat这个选择器)。

所以也能够说,运行时机制是多态的基础?~~~

23. 通知和协议的不一样之处?

答:协议有控制链(has-a)的关系,通知没有。

首先我一开始也不太明白,什么叫控制链(专业术语了~)。可是简单分析下通知和代理的行为模式,咱们大体能够有本身的理解

简单来讲,通知的话,它能够一对多,一条消息能够发送给多个消息接受者。

代理按咱们的理解,到不是直接说不能一对多,好比咱们知道的明星经济代理人,不少时候一个经济人负责好几个明星的事务。

只是对于不一样明星间,代理的事物对象都是不同的,一一对应,不可能说明天要处理A明星要一个发布会,代理人发出处理发布会的消息后,别称B的

发布会了。可是通知就不同,他只关心发出通知,而不关心多少接收到感兴趣要处理。

所以控制链(has-a从英语单词大体能够看出,单一拥有和可控制的对应关系。

24. 什么是推送消息?

答:推送通知更是一种技术。

简单点就是客户端获取资源的一种手段。

普通状况下,都是客户端主动的pull。

推送则是服务器端主动push。 测试push的实现能够查看该博文。

25. 关于多态性

答:多态,子类指针能够赋值给父类。

这个题目其实能够出到一切面向对象语言中,

所以关于多态,继承和封装基本最好都有个自我意识的理解,也并不是必定要把书上资料上写的能背出来

26. 对于单例的理解

答:在objective-c中要实现一个单例类,至少须要作如下四个步骤:

1).为单例对象实现一个静态实例,并初始化,而后设置成nil,

2).实现一个实例构造方法检查上面声明的静态实例是否为nil,若是是则新建并返回一个本类的实例,

3).重写allocWithZone方法,用来保证其余人直接使用alloc和init试图得到一个新实力的时候不产生一个新实例,

4).适当实现allocWitheZone,copyWithZone,release和autorelease。

27. 说说响应链

答: 事件响应链。包括点击事件,画面刷新事件等。在视图栈内从上至下,或者从下之上传播。

能够说点事件的分发,传递以及处理。具体能够去看下touch事件这块。由于问的太抽象化了

严重怀疑题目出到越后面就越笼统。

能够从责任链模式,来说经过事件响应链处理,其拥有的扩展性

28. frame和bounds有什么不一样?

答:frame指的是:该view在父view坐标系统中的位置和大小。(参照点是父亲的坐标系统)

bounds指的是:该view在自己坐标系统中 的位置和大小。(参照点是自己坐标系统)

29. 方法和选择器有何不一样?

答:selector是一个方法的名字,method是一个组合体,包含了名字和实现.

详情能够看apple文档。

30. OC的垃圾回收机制?

答: OC2.0有Garbage collection,可是iOS平台不提供。

通常咱们了解的objective-c对于内存管理都是手动操做的,可是也有自动释放池。

可是差了大部分资料,貌似不要和arc机制搞混就行了。

31. NSOperation queue?

答:存放NSOperation的集合类。

操做和操做队列,基本能够当作java中的线程和线程池的概念。用于处理ios多线程开发的问题。

网上部分资料提到一点是,虽然是queue,可是却并非带有队列的概念,放入的操做并不是是按照严格的先进现出。

这边又有个疑点是,对于队列来讲,先进先出的概念是Afunc添加进队列,Bfunc紧跟着也进入队列,Afunc先执行这个是必然的,

可是Bfunc是等Afunc彻底操做完之后,B才开始启动而且执行,所以队列的概念离乱上有点违背了多线程处理这个概念。

可是转念一想其实能够参考银行的取票和叫号系统。

所以对于A比B先排队取票可是B率先执行完操做,咱们亦然能够感性认为这仍是一个队列。

可是后来看到一票关于这操做队列话题的文章,其中有一句提到

“由于两个操做提交的时间间隔很近,线程池中的线程,谁先启动是不定的。”

瞬间以为这个queue名字有点忽悠人了,还不如pool~

综合一点,咱们知道他能够比较大的用处在于能够帮组多线程编程就行了。

32. 什么是延迟加载?

答:懒汉模式,只在用到的时候才去初始化。

也能够理解成延时加载。

我以为最好也最简单的一个列子就是tableView中图片的加载显示了。

一个延时载,避免内存太高,一个异步加载,避免线程堵塞。

33. 是否在一个视图控制器中嵌入两个tableview控制器?

答:一个视图控制只提供了一个View视图,理论上一个tableViewController也不能放吧,

只能说能够嵌入一个tableview视图。固然,题目自己也有歧义,若是不是咱们定性思惟认为的UIViewController,而是宏观的表示视图控制者,那咱们却是能够把其当作一个视图控制者,它能够控制多个视图控制器,好比TabbarController那样的感受。

34. 一个tableView是否能够关联两个不一样的数据源?你会怎么处理?

答:首先咱们从代码来看,数据源如何关联上的,实际上是在数据源关联的代理方法里实现的。

所以咱们并不关心如何去关联他,他怎么关联上,方法只是让我返回根据本身的须要去设置如相关的数据源。

所以,我以为能够设置多个数据源啊,可是有个问题是,你这是想干吗呢?想让列表如何显示,不一样的数据源分区块显示?

35. 何时使用NSMutableArray,何时使用NSArray?

答:当数组在程序运行时,须要不断变化的,使用NSMutableArray,当数组在初始化后,便再也不改变的,使用NSArray。须要指出的是,使用NSArray只代表的是该数组在运行时不发生改变,即不能往NSAarry的数组里新增和删除元素,但不代表其数组內的元素的内容不能发生改变。NSArray是线程安全的,NSMutableArray不是线程安全的,多线程使用到NSMutableArray须要注意。

36. 给出委托方法的实例,而且说出UITableVIew的Data Source方法

答:CocoaTouch框架中用到了大量委托,其中UITableViewDelegate就是委托机制的典型应用,是一个典型的使用委托来实现适配器模式,其中UITableViewDelegate协议是目标,tableview是适配器,实现UITableViewDelegate协议,并将自身设置为talbeview的delegate的对象,是被适配器,通常状况下该对象是UITableViewController。

UITableVIew的Data Source方法有- (NSInteger)tableView:(UITableView *)tableView numberOfRowsInSection:(NSInteger)section;

- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath;

37. 在应用中能够建立多少autorelease对象,是否有限制?

答案:无

38. 若是咱们不建立内存池,是否有内存池提供给咱们?

答:界面线程维护着本身的内存池,用户本身建立的数据线程,则须要建立该线程的内存池

39. 何时须要在程序中建立内存池?

答:用户本身建立的数据线程,则须要建立该线程的内存池

40. 类NSObject的那些方法常常被使用?

答:NSObject是Objetive-C的基类,其由NSObject类及一系列协议构成。

其中类方法alloc、class、 description 对象方法init、dealloc、– performSelector:withObject:afterDelay:等常常被使用

41. 什么是简便构造方法?

答:简便构造方法通常由CocoaTouch框架提供,如NSNumber的 + numberWithBool: + numberWithChar: + numberWithDouble: + numberWithFloat: + numberWithInt:

Foundation下大部分类均有简便构造方法,咱们能够经过简便构造方法,得到系统给咱们建立好的对象,而且不须要手动释放。

42. 如何使用Xcode设计通用应用?

答:使用MVC模式设计应用,其中Model层完成脱离界面,即在Model层,其是可运行在任何设备上,在controller层,根据iPhone与iPad(独有UISplitViewController)的不一样特色选择不一样的viewController对象。在View层,可根据现实要求,来设计,其中以xib文件设计时,其设置其为universal。

43. UIView的动画效果有那些?

答:有不少,如 UIViewAnimationOptionCurveEaseInOut UIViewAnimationOptionCurveEaseIn UIViewAnimationOptionCurveEaseOut UIViewAnimationOptionTransitionFlipFromLeft UIViewAnimationOptionTransitionFlipFromRight UIViewAnimationOptionTransitionCurlUpUIViewAnimationOptionTransitionCurlDown

44. 在iPhone应用中如何保存数据?

答:有如下几种保存机制:

1).经过web服务,保存在服务器上

2).经过NSCoder固化机制,将对象保存在文件中

3).经过SQlite或CoreData保存在文件数据库中

45. 什么是coredata?

答:coredata是苹果提供一套数据保存框架,其基于SQlite

46. 什么是NSManagedObject模型?

答:NSManagedObject是NSObject的子类 ,也是coredata的重要组成部分,它是一个通用的类,实现了core data 模型层所需的基本功能,用户可经过子类化NSManagedObject,创建本身的数据模型。

47. 什么是NSManagedobjectContext?

答:NSManagedobjectContext对象负责应用和数据库之间的交互。

48. 什么是谓词?

答:谓词是经过NSPredicate,是经过给定的逻辑条件做为约束条件,完成对数据的筛选。

1

2predicate = [NSPredicate predicateWithFormat:@"customerID == %d",n];

a = [customers filteredArrayUsingPredicate:predicate];

49. 和coredata一块儿有哪几种持久化存储机制?

答:存入到文件、 存入到NSUserDefaults(系统plist文件中)、存入到Sqlite文件数据库

50. 谈谈对Block 的理解?并写出一个使用Block执行UIVew动画?

答:Block是能够获取其余函数局部变量的匿名函数,其不但方便开发,而且能够大幅提升应用的执行效率(多核心CPU可直接处理Block指令)

1

2

3

4

5[UIView transitionWithView:self.view

duration:0.2

options:UIViewAnimationOptionTransitionFlipFromLeft

animations:^{ [[blueViewController view] removeFromSuperview]; [[self view] insertSubview:yellowViewController.view atIndex:0]; }

completion:NULL];

51. 写出上面代码的Block的定义。

答:

1

2typedef void(^animations) (void);

typedef void(^completion) (BOOL finished);

52. 试着使用+ beginAnimations:context:以及上述Block的定义,写出一个能够完成

1

+ (void)transitionWithView:(UIView *)view duration:(NSTimeInterval)duration options:(UIViewAnimationOptions)options animations:(void (^)(void))animations completion:(void (^)(BOOL finished))completion NS_AVAILABLE_IOS(4_0);

操做的函数执行部分

答案:无

53. 作过的项目是否涉及网络访问功能,使用什么对象完成网络功能?

答:ASIHTTPRequest与NSURLConnection

54. 简单介绍下NSURLConnection类及+ sendSynchronousRequest:returningResponse:error:与– initWithRequest:delegate:两个方法的区别?

答: NSURLConnection主要用于网络访问,其中+ sendSynchronousRequest:returningResponse:error:是同步访问数据,即当前线程会阻塞,并等待request的返回的response,而– initWithRequest:delegate:使用的是异步加载,当其完成网络访问后,会经过delegate回到主线程,并其委托的对象。

55. 多线程是什么

答: 多线程是个复杂的概念,按字面意思是同步完成多项任务,提升了资源的使用效率,从硬件、操做系统、应用软件不一样的角度去看,多线程被赋予不一样的内涵,对于硬件,如今市面上多数的CPU都是多核的,多核的CPU运算多线程更为出色;从操做系统角度,是多任务,如今用的主流操做系统都是多任务的,能够一边听歌、一边写博客;对于应用来讲,多线程可让应用有更快的回应,能够在网络下载时,同时响应用户的触摸操做。在iOS应用中,对多线程最初的理解,就是并发,它的含义是原来先作烧水,再摘菜,再炒菜的工做,会变成烧水的同时去摘菜,最后去炒菜。

56. iOS 中的多线程

答: iOS中的多线程,是Cocoa框架下的多线程,经过Cocoa的封装,可让咱们更为方便的使用线程,作过C++的同窗可能会对线程有更多的理解,好比线程的创立,信号量、共享变量有认识,Cocoa框架下会方便不少,它对线程作了封装,有些封装,可让咱们建立的对象,自己便拥有线程,也就是线程的对象化抽象,从而减小咱们的工程,提供程序的健壮性。

GCD是(Grand Central Dispatch)的缩写 ,从系统级别提供的一个易用地多线程类库,具备运行时的特色,能充分利用多核心硬件。GCD的API接口为C语言的函数,函数参数中多数有Block,关于Block的使用参看这里,为咱们提供强大的“接口”,对于GCD的使用参见本文

NSOperation与Queue

NSOperation是一个抽象类,它封装了线程的细节实现,咱们能够经过子类化该对象,加上NSQueue来同面向对象的思惟,管理多线程程序。具体可参看这里:一个基于NSOperation的多线程网络访问的项目。

NSThread

NSThread是一个控制线程执行的对象,它不如NSOperation抽象,经过它咱们能够方便的获得一个线程,并控制它。但NSThread的线程之间的并发控制,是须要咱们本身来控制的,能够经过NSCondition实现。

参看 iOS多线程编程之NSThread的使用

其余多线程

在Cocoa的框架下,通知、Timer和异步函数等都有使用多线程,(待补充).

57. 在项目何时选择使用GCD,何时选择NSOperation?

答: 项目中使用NSOperation的优势是NSOperation是对线程的高度抽象,在项目中使用它,会使项目的程序结构更好,子类化NSOperation的设计思路,是具备面向对象的优势(复用、封装),使得实现是多线程支持,而接口简单,建议在复杂项目中使用。

项目中使用GCD的优势是GCD自己很是简单、易用,对于不复杂的多线程操做,会节省代码量,而Block参数的使用,会是代码更为易读,建议在简单项目中使用。

58. 什么是block

答: 对于闭包(block),有不少定义,其中闭包就是可以读取其它函数内部变量的函数,这个定义即接近本质又较好理解。对于刚接触Block的同窗,会以为有些绕,由于咱们习惯写这样的程序main(){ funA();} funA(){funB();} funB(){…..}; 就是函数main调用函数A,函数A调用函数B… 函数们依次顺序执行,但现实中不全是这样的,例如项目经理M,手下有3个程序员A、B、C,当他给程序员A安排实现功能F1时,他并不等着A完成以后,再去安排B去实现F2,而是安排给A功能F1,B功能F2,C功能F3,而后可能去写技术文档,而当A遇到问题时,他会来找项目经理M,当B作完时,会通知M,这就是一个异步执行的例子。在这种情形下,Block即可大显身手,由于在项目经理M,给A安排工做时,同时会告诉A若果遇到困难,如何能找到他报告问题(例如打他手机号),这就是项目经理M给A的一个回调接口,要回掉的操做,好比接到电话,百度查询后,返回网页内容给A,这就是一个Block,在M交待工做时,已经定义好,而且取得了F1的任务号(局部变量),倒是在当A遇到问题时,才调用执行,跨函数在项目经理M查询百度,得到结果后回调该block。

59. block 实现原理

答: Objective-C是对C语言的扩展,block的实现是基于指针和函数指针。

从计算语言的发展,最先的goto,高级语言的指针,到面向对象语言的block,从机器的思惟,一步步接近人的思惟,以方便开发人员更为高效、直接的描述出现实的逻辑(需求)。

使用实例

cocoaTouch框架下动画效果的Block的调用

使用typed声明block

1

2typedef void(^didFinishBlock) (NSObject *ob);

这就声明了一个didFinishBlock类型的block,

而后即可用

1

@property (nonatomic,copy) didFinishBlock finishBlock;

声明一个blokc对象,注意对象属性设置为copy,接到block 参数时,便会自动复制一份。

__block是一种特殊类型,

使用该关键字声明的局部变量,能够被block所改变,而且其在原函数中的值会被改变。

60.关于block

答: 面试时,面试官会先问一些,是否了解block,是否使用过block,这些问题至关于开场白,每每是下面一系列问题的开始,因此必定要如实根据本身的状况回答。

1). 使用block和使用delegate完成委托模式有什么优势?

首先要了解什么是委托模式,委托模式在iOS中大量应用,其在设计模式中是适配器模式中的对象适配器,Objective-C中使用id类型指向一切对象,使委托模式更为简洁。了解委托模式的细节:

iOS设计模式—-委托模式

使用block实现委托模式,其优势是回调的block代码块定义在委托对象函数内部,使代码更为紧凑;

适配对象再也不须要实现具体某个protocol,代码更为简洁。

2). 多线程与block

GCD与Block

使用 dispatch_async 系列方法,能够以指定的方式执行block

GCD编程实例

dispatch_async的完整定义

1

2

3void dispatch_async(

dispatch_queue_t queue,

dispatch_block_t block);

功能:在指定的队列里提交一个异步执行的block,不阻塞当前线程

经过queue来控制block执行的线程。主线程执行前文定义的 finishBlock对象

1

dispatch_async(dispatch_get_main_queue(),^(void){finishBlock();});

62.谈谈Object-C的内存管理方式及过程?

答: 1).当你使用new,alloc和copy方法建立一个对象时,该对象的保留计数器值为1.当你再也不使用该对象时,你要负责向该对象发送一条release或autorelease消息.这样,该对象将在使用寿命结束时被销毁.

2).当你经过任何其余方法得到一个对象时,则假设该对象的保留计数器值为1,并且已经被设置为自动释放,你不须要执行任何操做来确保该对象被清理.若是你打算在一段时间内拥有该对象,则须要保留它并确保在操做完成时释放它.

3).若是你保留了某个对象,你须要(最终)释放或自动释放该对象.必须保持retain方法和release方法的使用次数相等.

63.Object-C有私有方法吗?私有变量呢?

答: objective-c – 类里面的方法只有两种, 静态方法和实例方法. 这彷佛就不是完整的面向对象了,按照OO的原则就是一个对象只暴露有用的东西. 若是没有了私有方法的话, 对于一些小范围的代码重用就不那么顺手了. 在类里面声名一个私有方法

1

2

3

4

5

6

7@interfaceController : NSObject { NSString *something; }

+ (void)thisIsAStaticMethod;

– (void)thisIsAnInstanceMethod;

@end

@interfaceController (private) -

(void)thisIsAPrivateMethod;

@end

@private能够用来修饰私有变量

在Objective‐C中,全部实例变量默认都是私有的,全部实例方法默认都是公有的

64.Object-C有多继承吗?没有的话用什么代替?cocoa 中全部的类都是NSObject 的子类

答: 多继承在这里是用protocol 委托代理 来实现的

你不用去考虑繁琐的多继承 ,虚基类的概念.

ood的多态特性 在 obj-c 中经过委托来实现.

65.内存管理 Autorelease、retain、copy、assign的set方法和含义?

答: 1).你初始化(alloc/init)的对象,你须要释放(release)它。例如:

NSMutableArray aArray = [[NSArray alloc] init]; 后,须要 [aArray release];

2).你retain或copy的,你须要释放它。例如:

[aArray retain] 后,须要 [aArray release];

3).被传递(assign)的对象,你须要斟酌的retain和release。例如:

obj2 = [[obj1 someMethod] autorelease];

对象2接收对象1的一个自动释放的值,或传递一个基本数据类型(NSInteger,NSString)时:你或但愿将对象2进行retain,以防止它在被使用以前就被自动释放掉。可是在retain后,必定要在适当的时候进行释放。

关于索引计数(Reference Counting)的问题

retain值 = 索引计数(Reference Counting)

NSArray对象会retain(retain值加一)任何数组中的对象。当NSArray被卸载(dealloc)的时候,全部数组中的对象会 被 执行一次释放(retain值减一)。不只仅是NSArray,任何收集类(Collection Classes)都执行相似操做。例如 NSDictionary,甚至UINavigationController。

Alloc/init创建的对象,索引计数为1。无需将其再次retain。

[NSArray array]和[NSDate date]等“方法”创建一个索引计数为1的对象,可是也是一个自动释放对象。因此是本地临时对象,那么无所谓了。若是是打算在全Class中使用的变量(iVar),则必须retain它。

缺省的类方法返回值都被执行了“自动释放”方法。(*如上中的NSArray)

在类中的卸载方法“dealloc”中,release全部未被平衡的NS对象。(*全部未被autorelease,而retain值为1的)

66. C和obj-c 如何混用

答: 1).obj-c的编译器处理后缀为m的文件时,能够识别obj-c和c的代码,处理mm文件能够识别obj-c,c,c++代码,但cpp文件必须只能用c/c++代码,并且cpp文件include的头文件中,也不能出现obj-c的代码,由于cpp只是cpp

2).在mm文件中混用cpp直接使用便可,因此obj-c混cpp不是问题

3).在cpp中混用obj-c其实就是使用obj-c编写的模块是咱们想要的。

若是模块以类实现,那么要按照cpp class的标准写类的定义,头文件中不能出现obj-c的东西,包括#import cocoa的。实现文件中,即类的实现代码中可使用obj-c的东西,能够import,只是后缀是mm。

若是模块以函数实现,那么头文件要按c的格式声明函数,实现文件中,c++函数内部能够用obj-c,但后缀仍是mm或m。

总结:只要cpp文件和cpp include的文件中不包含obj-c的东西就能够用了,cpp混用obj-c的关键是使用接口,而不能直接使用 实现代 码,实际上cpp混用的是obj-c编译后的o文件,这个东西实际上是无差异的,因此能够用。obj-c的编译器支持cpp

67. Objective-C堆和栈的区别?

答: 管理方式:对于栈来说,是由编译器自动管理,无需咱们手工控制;对于堆来讲,释放工做由程序员控制,容易产生memory leak。

申请大小:

栈:在Windows下,栈是向低地址扩展的数据结构,是一块连续的内存的区域。这句话的意思是栈顶的地址和栈的最大容量是系统预先规定好的,在 WINDOWS下,栈的大小是2M(也有的说是1M,总之是一个编译时就肯定的常数),若是申请的空间超过栈的剩余空间时,将提示overflow。因 此,能从栈得到的空间较小。

堆:堆是向高地址扩展的数据结构,是不连续的内存区域。这是因为系统是用链表来存储的空闲内存地址的,天然是不连续的,而链表的遍历方向是由低地址向高地址。堆的大小受限于计算机系统中有效的虚拟内存。因而可知,堆得到的空间比较灵活,也比较大。

碎片问题:对于堆来说,频繁的new/delete势必会形成内存空间的不连续,从而形成大量的碎片,使程序效率下降。对于栈来说,则不会存在这个问题,由于栈是先进后出的队列,他们是如此的一一对应,以致于永远都不可能有一个内存块从栈中间弹出

分配方式:堆都是动态分配的,没有静态分配的堆。栈有2种分配方式:静态分配和动态分配。静态分配是编译器完成的,好比局部变量的分配。动态分配由alloca函数进行分配,可是栈的动态分配和堆是不一样的,他的动态分配是由编译器进行释放,无需咱们手工实现。

分配效率:栈是机器系统提供的数据结构,计算机会在底层对栈提供支持:分配专门的寄存器存放栈的地址,压栈出栈都有专门的指令执行,这就决定了栈的效率比较高。堆则是C/C++函数库提供的,它的机制是很复杂的。

68. ViewController的didReceiveMemoryWarning怎么被调用:

答:[supper didReceiveMemoryWarning];

69.何时用delegate,何时用Notification?

答: delegate针对one-to-one关系,用于sender接受到reciever的某个功能反馈值。

notification针对one-to-one/many/none,reciver,用于通知多个object某个事件。

70.用预处理指令#define声明一个常数,用以代表1年中有多少秒(忽略闰年问题)

答:

#define SECONDS_PER_YEAR (60 * 60 * 24 * 365)UL

我在这想看到几件事情:

#define 语法的基本知识(例如:不能以分号结束,括号的使用,等等)

懂得预处理器将为你计算常数表达式的值,所以,直接写出你是如何计算一年中有多少秒而不是计算出实际的值,是更清晰而没有代价的。

意识到这个表达式将使一个16位机的整型数溢出-所以要用到长整型符号L,告诉编译器这个常数是的长整型数。

若是你在你的表达式中用到UL(表示无符号长整型),那么你有了一个好的起点。记住,第一印象很重要。

71.写一个”标准"宏MIN ,这个宏输入两个参数并返回较小的一个。

答:

1

#define MIN(A,B) ((A) <= (B) ? (A) : (B))

这个测试是为下面的目的而设的:

标识#define在宏中应用的基本知识。这是很重要的,由于直到嵌入(inline)操做符变为标准C的一部分,宏是方便产生嵌入代码的惟一方

法,

对于嵌入式系统来讲,为了能达到要求的性能,嵌入代码常常是必须的方法。

三重条件操做符的知识。这个操做符存在C语言中的缘由是它使得编译器能产生比 if-then-else 更优化的代码,了解这个用法是很重要的。

懂得在宏中当心地把参数用括号括起来

我也用这个问题开始讨论宏的反作用,例如:当你写下面的代码时会发生什么事?

1

least = MIN(*p++, b);

结果是:

1

((*p++) <= (b) ? (*p++) : (*p++))

这个表达式会产生反作用,指针p会做三次++自增操做。

72.关键字const有什么含意?修饰类呢?static的做用,用于类呢?还有extern c的做用

答:

const 意味着"只读",下面的声明都是什么意思?

1

2

3

4

5const int a;

int const a;

const int *a;

int * const a;

int const * a const;

前两个的做用是同样,a是一个常整型数。

第三个意味着a是一个指向常整型数的指针(也就是,整型数是不可修改的,但指针能够)。

第四个意思a是一个指向整型数的常指针(也就是说,指针指向的整型数是能够修改的,但指针是不可修改的)。

最后一个意味着a是一个指向常整型数的常指针(也就是说,指针指向的整型数是不可修改的,同时指针也是不可修改的)。

结论:

关键字const的做用是为给读你代码的人传达很是有用的信息,实际上,声明一个参数为常量是为了告诉了用户这个参数的应用目的。

若是你曾花不少时间清理其它人留下的垃圾,你就会很快学会感谢这点多余的信息。(固然,懂得用const的程序员不多会留下的垃圾让别人来清理的)  经过给优化器一些附加的信息,使用关键字const也许能产生更紧凑的代码。合理地使用关键字const可使编译器很天然地保护那些不但愿被改变的参数,防止其被无心的代码修改。简而言之,这样能够减小bug的出现。

1).欲阻止一个变量被改变,可使用 const 关键字。在定义该 const 变量时,一般须要对它进行初

始化,由于之后就没有机会再去改变它了;

2).对指针来讲,能够指定指针自己为 const,也能够指定指针所指的数据为 const,或两者同时指

定为 const;

3).在一个函数声明中,const 能够修饰形参,代表它是一个输入参数,在函数内部不能改变其值;

4).对于类的成员函数,若指定其为 const 类型,则代表其是一个常函数,不能修改类的成员变量;

5).对于类的成员函数,有时候必须指定其返回值为 const 类型,以使得其返回值不为“左值”。

73. 关键字volatile有什么含意?并给出三个不一样的例子。

答:一个定义为 volatile的变量是说这变量可能会被意想不到地改变,这样,编译器就不会去假设这个变量的值了。精确地说就是,优化器在用到这个变量时必须每次都当心地从新读取这个变量的值,而不是使用保存在寄存器里的备份。

下面是volatile变量的几个例子:

并行设备的硬件寄存器(如:状态寄存器)

一个中断服务子程序中会访问到的非自动变量(Non-automatic variables)

多线程应用中被几个任务共享的变量

74. 一个参数既能够是const还能够是volatile吗? 一个指针能够是volatile 吗?解释为何。

答:1).是的。一个例子是只读的状态寄存器。它是volatile由于它可能被意想不到地改变。它是const由于程序不该该试图去修改它。

2).是的。尽管这并不很常见。一个例子是当一个中服务子程序修该一个指向一个buffer的指针时。

75 . static 关键字的做用:

答:

1).函数体内 static 变量的做用范围为该函数体,不一样于 auto 变量,该变量的内存只被分配一次,

所以其值在下次调用时仍维持上次的值;

2).在模块内的 static 全局变量能够被模块内所用函数访问,但不能被模块外其它函数访问;

3).在模块内的 static 函数只可被这一模块内的其它函数调用,这个函数的使用范围被限制在声明

它的模块内;

4).在类中的 static 成员变量属于整个类所拥有,对类的全部对象只有一份拷贝;

5).在类中的 static 成员函数属于整个类所拥有,这个函数不接收 this 指针,于是只能访问类的static 成员变量。

76. 线程与进程的区别和联系?

答:

1). 进程和线程都是由操做系统所体会的程序运行的基本单元,系统利用该基本单元实现系统对应用的并发性

2). 进程和线程的主要差异在于它们是不一样的操做系统资源管理方式。

3). 进程有独立的地址空间,一个进程崩溃后,在保护模式下不会对其它进程产生影响,而线程只是一个进程中的不一样执行路径。

4.)线程有本身的堆栈和局部变量,但线程之间没有单独的地址空间,一个线程死掉就等于整个进程死掉。因此多进程的程序要比多线程的程序健壮,但在进程切换时,耗费资源较大,效率要差一些。

5). 但对于一些要求同时进行而且又要共享某些变量的并发操做,只能用线程,不能用进程。

77. 列举几种进程的同步机制,并比较其优缺点。

答: 原子操做  信号量机制    自旋锁    管程,会合,分布式系统

78. 进程之间通讯的途径

答:共享存储系统消息传递系统管道:以文件系统为基础

79. 进程死锁的缘由

答:资源竞争及进程推动顺序非法