iOS面试题

一个组别渡过好死的面试题

考察一个面试者基础咋样,基本上问一个 @property 就足够了:

@property 后面可以来什么修饰符?

  • 线程安全的:
    • atomic,nonatomic
  • 拜权限的
    • readonly,readwrite
  • 内存管理(ARC)
    • assign,strong,weak,copy
  • 内存管理(MRC)
  • assign,retain,copy
  • 点名方法名称
  • setter=
  • getter=

咦情况使用 weak 关键字,相比 assign 有啊不同?比如:

  • 每当ARC中,出现循环引用的时光,必须要来一致端采用weak,比如:自定义View的代办属性
  • 一度自己都指向它们进行相同坏强使,没有必要当强引用一次,此时啊会以weak,自定义View的支行控件属性一般也以weak;但b是为得以用strong
  • weak当对象销毁之时节,指针会让机关安装为nil,而assign不会见* assigin
    可以据此非OC对象,而weak必须用于OC对象

怎么用 copy 关键字?

  • 对此字符串和block的性能一般下copy
  • 字符串使用copy是为外部把字符串内容改动了,影响该属性
  • block用copy是以MRC遗留下来的,在MRC中,方法中的block是于当栈区底,使用copy可以管它们内置堆区.在ACR中对于block使用copy还是strong效果是同的

以此写法会发生什么问题: @property (copy) NSMutableArray *array;

  • 添加,删除,修改数组内的因素的时段,程序会为找不至于的方要崩溃.因为copy就是复制一个不可变NSArray的对象

何以被祥和之类用 copy 修饰符?

  • 而是说为自家的类为支持copy的功力也?
  • 假使面试官说是:
    • 遵守NSCopying协议
    • 实现 – (id)copyWithZone:(NSZone *)zone; 方法
  • 要是面试官说否,是性质被什么使用copy
    • 每当采取字符串和block的时光一般都用copy

何以还写带 copy 关键字的 setter?

  • 双重写copy的setter方法时候,一定要调用一下传唱的对象的copy方法,然后于赋值给该setter的方式对应之成员变量

及时无异于套问题区分度比较充分,如果地方的问题且能够应对是,可以拉开问还深入点的:

@property 的真相是什么?ivar、getter、setter 是怎么变化并加加到此看似中之
  • 每当平凡的OC对象吃,@property就是编译其自动帮咱转移一个私房的积极分子变量和setter与getter方法的宣示和促成
  • 本身为来清属性是怎落实之,曾经反编译过相关的代码,他大致生成了五个个东西
  • OBJC_IVAR_$类名$特性名称 该属性之偏移量
  • setter与getter方法对应的贯彻函数
  • ivar_list 就是成员变量列表
  • method_list 方法列表
  • prop_list 属性列表
    也就是说我们每次在搭一个性能,系统还见面于ivar_list中补充加一个分子变量的描述,在method_list中加进setter与getter方法的叙说,在性质列表中增一个性能的性之描述,然后计算该属性在目标被之偏移量,然后伸出setter与getter方法对应之落实,在setter方法方法被起偏移量的职位上马赋值,在getter方法中从偏移量开始取值,为了能读取正确字节数,系统对象偏移量的指针类型进行了档次强转.
@protocol 和 category 中哪些以 @property
  1. 以protocol中采用property只见面生成setter和getter方法声明,我们使用性能的目的,是期望遵守我商量的靶子的贯彻该属性
  2. category 用 @property
    也是止见面生成setter和getter方法的扬言,如果我们真的用给category增加属性之兑现,需要靠运行时的鲜个函数

    1. objc_setAssociatedObject
    2. objc_getAssociatedObject
runtime 如何落实 weak 属性

runtime 对注册的接近, 会进行布局,对于 weak 对象见面放入一个 hash 表中。 用
weak 指向的靶子地址作为 key,当此对象的援计数为0的下会 dealloc,
进而在是 weak 表中找到这个目标地址为键的持有 weak 对象,从而设置为 nil

每个人擅长的圈子不一样,我们一般会于简历及搜索好写擅长的艺聊,假如自己连无是老熟,最好别写出来或扯出来,万一面试官刚好非常会这里虽露馅了。

Checklist

总过数对试题,没坚持下去,后来将这些当
checklist,面试的时段实在没话聊的时候召开只提醒,语言、框架、运行机制性质的:

[※]@property中生出哪性关键字?

同上

[※]weak属性需要以dealloc中置nil么?

不需要,在ARC环境无论是强指针还是弱指针都无需在deallco设置为nil,ARC会自动帮我们处理

[※※]@synthesize和@dynamic分别发出啊作用?

  1. @property有个别只照应的乐章,一个是@synthesize,一个是@dynamic。如果@synthesize和@dynamic都并未写,那么默认的哪怕是@syntheszie
    var = _var;
  2. @synthesize的语义是一旦您从未手动实现setter方法和getter方法,那么编译器会活动为您长这半单方法。
  3. @dynamic告诉编译器,属性的setter与getter方法由用户自己实现,不自动生成。(当然对readonly的性只需要提供getter即可)。假如一个特性让声称也@dynamic
    var,然后你未曾供@setter方法和@getter方法,编译的当儿没有问题,但是当程序运行到instance.var
    =someVar,由于缺setter方法会招致程序崩溃;或者当运行及 someVar =
    var时,由于缺getter方法同样会造成崩溃。编译时不曾问题,运行时才实施相应的不二法门,这就是是所谓的动态绑定。

[※※※]ARC下,不亮指定其他性质关键字时,默认的重大字都起怎样?

  1. 本着承诺着力数据列默认关键字是
    atomic,readwrite,assign
  2. 对常见的OC对象
    atomic,readwrite,strong

[※※※]因而@property声明的NSString(或NSArray,NSDictionary)经常应用copy关键字,为什么?如果改用strong关键字,可能造成什么问题?

  1. 为父类指针可以针对子类对象,使用copy的目的是为了让本对象的性不让外界影响,使用copy无论给自身传入是一个可变对象或不行对象,我自有所的即使是一个不可变的抱本.
  2. 而我们应用是strong,那么是特性就发出或指于一个可变对象,如果这个可变对象在外表为改了,那么会潜移默化该属性.

[※※※]@synthesize合成实例变量的规则是啊?假如property名为foo,存在一个名也_foo的实例变量,那么还会自动合成新换量么?

要是无点名成员变量的名称及自动生成一个特性同名的积极分子变量,如果指定的成员变量的称呼,会转一个指定的称谓的成员变量,如果这成员曾在了即不再深成了.
倘是 @synthesize foo; 还会转移一个称为foo的成员变量
如果是 @synthesize foo = _foo; 就不见面生成成员变量了.

[※※※※※]于有矣自动合成属性实例变量之后,@synthesize还有什么使用状况?

@synthesize主要就之所以来生成setter,getter方法的实现,在@property被增长后,其实已经老少动@synthesize了,你懂@synthesize的外以状况吧?
能于本人介绍一下也?
若你听明白了,感觉面试官说的万分有道理,可以说接触许的话.

[※※]objc中向一个nil对象发送信息将会晤时有发生啊?

  1. 每当Objective-C中于nil发送信息是截然可行之——只是以运作时不见面生其它作用:
    • 假若一个方返回值是一个目标,那么发送给nil的音将回回0(nil)。例如:Person *
      motherInlaw = [ aPerson spouse]
      mother];如果spouse对象呢nil,那么发送给nil的信息mother也拿回nil。
    • 倘措施返回值为指针类型,其指针大小也小于或者当sizeof(void*),float,double,long
      double 或者long long的整型标量,发送给nil的音信将返回回0。
    • 倘方式返回值为结构体,发送给nil的信将回到回0。结构体中逐条字段的价值将还是0。
    • 如果方式的归值未是上述涉的几乎种情形,那么发送给nil的音信的归来值将是无定义的。

[※※※]objc中向一个目标发送信息[obj foo]和objc_msgSend()函数之间发生什么关联?

该方式编译之后便是objc_msgSend()函数调用.如果我没记错的光景是这么的.
((void (*)(id, SEL))(void *)objc_msgSend)((id)obj,
sel_registerName(“foo”));

[※※※]啊时候会报unrecognized selector的死?

  • 当该目标及有方法,而该对象及从来不落实者方式的时候

[※※※※]一个objc对象如何开展内存布局?(考虑有父类的情)

  • 备父类的积极分子变量和协调的分子变量都见面存放于该对象所对应之仓储空间中.
  • 每一个目标中还一个isA指针,指于外的类似对象,类对象被存放着以对象的对象方法列表和分子变量的列表,属性列表,它里面也来一个isA指针指向元对象(meta
    class),元对象中存放的是相近措施列表,类对象中还有一个superclass的指针,指为外的父类对象
  • 彻底对象就是是NSobject
  • 如图:
![](https://www.jianshu.com/pF7C043-5B26-4E33-A117-B7D35008D663.png)

[※※※※]一个objc对象的isa的指针指于什么?有什么打算?

  • 凭借于他的类对象,从而得以查找到对象上的方

[※※※※]下面的代码输出什么?

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

出口的结果还是:Son,
由:super 和 self 都是借助于的本实例对象的,
不等之凡,super调用之跳过本类方法,调用父类的办法
父类方法的class方法自然还是当基类中实现之,所以管使用self和super调用都是平的.
具体分析参照刨根问底Objective-C Runtime(1)- Self &
Super

[※※※※]runtime如何通过selector找到呼应的IMP地址?(分别考虑类方式和实例方法)

  • 各国一个类似对象被还一个法列表,方法列表中著录在法子的称号,方法实现,以及参数类型,其实selector本质就是办法名称,通过之艺术名称即使足以以方式列表中找到呼应的艺术实现.

[※※※※]利用runtime Associate方法关联的对象,需要以主对象dealloc的时释放么?

  • 当ARC下未需
  • 以MRC中,对于下retain或copy策略的内需

[※※※※※]objc中的类措施及实例方法来啊本质区别和沟通?

  • 类方法
    • 恍如方式是属类对象的
    • 仿佛方式就会由此类似对象调用
    • 类方式中的self是相仿对象
    • 恍如方式可调用其他的类似方式
    • 接近措施被未克顾成员变量
    • 类措施吃乱直接调用对象方法
  • 实例方法
    • 实例方法是属实例对象的
    • 实例方法只有会经过实例对象调用
    • 实例方法被之self是实例对象
    • 实例方法吃可以看成员变量
    • 实例方法中一直调用实例方法
    • 实例方法被呢得以调用类方法(通过类名)

[※※※※※]_objc_msgForward函数是举行啊的,直接调用它将会见产生什么?

  1. 尚未哟研究过,从名字来拘禁是用来转发信息之,你可知于自己讲话出口吧?谢谢!

[※※※※※]runtime如何贯彻weak变量的全自动置nil?

1. 没有研究过,你有研究过吗,可以给我讲讲吗?
2. 我猜系统会维护一个弱指针列表,当某个对象销毁时候,它会把所有指向该对象的弱指针设置为nil

[※※※※※]可否为编译后得到的好像吃追加实例变量?能否为运行时创造的接近吃补充加实例变量?为什么?

    因为编译后的类已经注册在 runtime 中,类结构体中的 objc_ivar_list 实例变量的链表 和 instance_size 实例变量的内存大小已经确定,同时runtime 会调用 class_setIvarLayout 或 class_setWeakIvarLayout 来处理 strong weak 引用。所以不能向存在的类中添加实例变量,
运行时创建的类是可以添加实例变量,调用 class_addIvar 函数。但是得在调用 objc_allocateClassPair 之后,objc_registerClassPair 之前,原因同上。

[※※※]runloop和线程有啊关系?

 1. 每一个线程中都一个runloop,只有主线的的runloop默认是开启的,其他线程的runloop是默认没有开启的
 2. 可以通过CFRunLoopRun() 函数来开启一个事件循环
 3. 看SDWebImage源码的时候见到有这么用过.

[※※※]runloop的mode作用是啊?

model 主要是用来指定时间以运行循环中的先级的
苹果公开提供的 Mode 有有限单:
kCFRunLoopDefaultMode
kCFRunLoopCommonModes

假定我们管一个NSTimer对象为kCFRunLoopDefaultMode添加到主运行循环中的当儿,当一直有用户事件处理的时,NSTimer将不再给调度
如我们将一个NSTimer对象为kCFRunLoopCommonModes添加到主运行循环中的时刻,当一直闹用户事件处理的早晚,NSTimer还能正常的调度,互不影响.

[※※※※]因+ scheduledTimerWithTimeInterval…的方触发的timer,在滑行页面及之列表时,timer会暂定回调,为什么?如何缓解?

同上

[※※※※※]猜想runloop内部是什么贯彻之?

  1. 他是一个死循环
  2.如果事件队列中存放在事件,那就取出事件,执行相关代码
  3.如果没有事件,就挂起,等有事件了,立即唤醒事件循环,开始执行.

 简单来说。。。
function loop() {
    initialize();
    do {
        var message = get_next_message();
        process_message(message);
    } while (message != quit);
} 

[※]objc以什么机制管理对象内存?

* MRC 手动引用计数
* ARC 自动引用计数,现在通常使用自动引用计数

[※※※※]ARC通过什么法帮助开发者管理内存?

通过编译器在编译的时候,插入如内管理的代码

[※※※※]莫手动指定autoreleasepool的前提下,一个autorealese对象在啊时刻释放?(比如当一个vc的viewDidLoad中创造)

在每次事件循环开始创建自动释放池,在每次事件结束销毁自动释放池
以viewDidLoad方法为例,可以理解为在viewDidLoad方法开始执行之前创建自动释放池,
在viewDidLoad方法执行之后销毁自动释放吃

[※※※※]BAD_ACCESS在什么动静下出现?

1.  死循环了
2.  访问一个僵尸对象

[※※※※※]苹果是哪兑现autoreleasepool的?

1. 我猜想autoreleasepool 本质就是一个队列(数组),
2. 当调用autorelease的时候会把该对象添加到autoreleasepool中,并且把引用计数+1
3. 当autoreleasepool即将销毁的时候,把其中的所有对象进行一次release操作

[※※]运block时什么状态会发生引用循环,如何解决?

 只要是一个对象对该block进行了强引用,在block内部有直接使用到该对象,

[※※]在block内什么改block外部变量?

  1. 通过 __bock修改的表变量,可以在block内部修改
  2. 怀念装B的语句可以说一下__bock内部举行了什么事

[※※※]动用系统的一些block api(如UIView的block版本写动画时),是否为考虑引用循环问题?

一般不用考虑,因为官方文档中没有告诉我们要注意发生强引用,所以推测系统控件一般没有对这些block进行强引用,所以我们可以不用考虑循环强引用的问题

[※※]GCD的队列(dispatch_queue_t)分哪点儿种植档次?

  串行队列和并行队列

[※※※※]如何用GCD同步若干独异步调用?(如基于若干个url异步加载多布置图纸,然后在还下充斥完成后合成一张整图)

  总体上说:  使用 dispatch group,然后 wait forever 等待完成, 或者采取 group notify 来通知回调。
  细节:
  1. 创建异步队列
  2. 创建dispatch_group  dispatch_group_t =  dispatch_group_create()
  3. 通过组来执行异步下载任务
    dispatch_group_async(queueGroup, aQueue, ^{    
          NSLog(@"下载图片.");    
    }); 
  4.等到所有任务完成   dispatch_group_wait(queueGroup, DISPATCH_TIME_FOREVER);
  5.合成图片

[※※※※]dispatch_barrier_async的作用是什么?

barrier:是障碍物的意思,在多个并行任务中间,他就像是一个隔离带,把前后的并行任务分开.
dispatch_barrier_async 作用是在并行队列中,等待前面操作并行任务完成再执行dispatch_barrier_async中的任务,如果后面还有并行任务,会开始执行后续的并行任务
[※※※※※]苹果为什么要废弃dispatch_get_current_queue?
容易误用造成死锁

[※※※※※]以下代码运行结果如何?

- (void)viewDidLoad
{
    [super viewDidLoad];
    NSLog(@"1");
    dispatch_sync(dispatch_get_main_queue(), ^{
        NSLog(@"2");
    });
    NSLog(@"3");
}
  1. 止能够出口1,然后线程主线程死锁

[※※]addObserver:forKeyPath:options:context:各个参数的用意分别是什么,observer中得实现谁方法才会博得KVO回调?

   // 添加键值观察
    /**
     1. 调用对象:要监听的对象
     2. 参数
     1> 观察者,负责处理监听事件的对象
     2> 观察的属性
     3> 观察的选项
     4> 上下文
     */
    [self.person addObserver:self forKeyPath:@"name" options:NSKeyValueObservingOptionNew | NSKeyValueObservingOptionOld context:@"Person Name"];

 // NSObject 分类方法,意味着所有的 NSObject 都可以实现这个方法!
// 跟协议的方法很像,分类方法又可以称为“隐式代理”!不提倡用,但是要知道概念!
// 所有的 kvo 监听到事件,都会调用此方法
/**
 1. 观察的属性
 2. 观察的对象
 3. change 属性变化字典(新/旧)
 4. 上下文,与监听的时候传递的一致

 可以利用上下文区分不同的监听!
 */
- (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary *)change context:(void *)context {

    NSLog(@"睡会 %@", [NSThread currentThread]);

    [NSThread sleepForTimeInterval:1.0];

    NSLog(@"%@ %@ %@ %@", keyPath, object, change, context);
}   

[※※※]怎样手动触发一个value的KVO

1.通过setValue:forKey: 给属性赋值
2.通过setValue:forKeyPath: 给属性赋值
3.直接调用setter方法方法给属性赋值
4.直接通过指针给属性赋值
5.

给这个value设置一个值,就可以触发了

[※※※]假如一个好像有实例变量NSString *_foo,调用setValue:forKey:时,可以以foo还是_foo作为key?

 都可以

[※※※※]KVC的keyPath中的汇运算符如何以?

1. 必须用在集合对象上或普通对象的集合属性上
2. 简单集合运算符有@avg, @count , @max , @min ,@sum,
3. 格式  @"@sum.age"或 @"集合属性.@max.age"  

[※※※※]KVC和KVO的keyPath一定是属性么?

1.一个可以是成员变量

[※※※※※]争关闭默认的KVO的默认实现,并登从定义的KVO实现?

什么样协调动手实现
KVO

[※※※※※]apple用什么措施实现对一个对象的KVO?

 同上

[※※]IBOutlet连出来的视图属性为什么可以叫设置成weak?

因为视图已经对它有一个强引用了

[※※※※※]IB中User Defined Runtime Attributes如何运用?

    User Defined Runtime Attributes 是一个不被看重但功能非常强大的的特性,
    它能够通过KVC的方式配置一些你在interface builder 中不能配置的属性。当你希望在IB中作尽可能多得事情,
    这个特性能够帮助你编写更加轻量级的viewcontroller

[※※※]争调试BAD_ACCESS错误

 1.设置全局断点快速定位问题代码所在行

[※※※]lldb(gdb)常用的调节命令?

 最常用就是 : po 对象

外的参照
浅谈LLDB调试器

相关文章