iOS每日面试题一

525 阅读3分钟

1、如何令自己所写的对象具有拷贝功能?

OC的话如果想让自己的类具备copy方法,并返回不可变类型,必须遵循nscopying协议,并且实现- (id)copyWithZone:(NSZone *)zone; 如果让自己的类具备mutableCopy方法,并且返回可变类型,必须遵守NSMutableCopying,并实现- (id)mutableCopyWithZone:(nullable NSZone *)zone

Swift类拷贝:

func copyDishViewModel() -> DishViewModel {
    let copyDishViewModel = DishViewModel(withDishModel: self.dishModel)
    copyDishViewModel.tips = self.tips
    copyDishViewModel.labels = self.labels.map{ $0.copyDishLabel() }
 }

2、说说你理解weak属性?

Runtime维护了一个weak表,用于存储指向某个对象的所有weak指针。weak表其实是一个hash(哈希)表,Key是所指对象的地址,Value是weak指针的地址(这个地址的值是所指对象的地址)数组。

1、初始化时:runtime会调用objc_initWeak函数,初始化一个新的weak指针指向对象的地址。

2、添加引用时:objc_initWeak函数会调用 objc_storeWeak() 函数, objc_storeWeak() 的作用是更新指针指向,创建对应的弱引用表。

3、释放时,调用clearDeallocating函数。clearDeallocating函数首先根据对象地址获取所有weak指针地址的数组,然后遍历这个数组把其中的数据设为nil,最后把这个entry从weak表中删除,最后清理对象的记录。

追问的问题一:

1.实现weak后,为什么对象释放后会自动为nil?

runtime 对注册的类, 会进行布局,对于 weak 对象会放入一个 hash 表中。 用 weak 指向的对象内存地址作为 key,当此对象的引用计数为 0 的时候会 dealloc,假如 weak 指向的对象内存地址是 a ,那么就会以 a 为键, 在这个 weak 表中搜索,找到所有以 a 为键的 weak 对象,从而设置为 nil 。

追问的问题二:

2.当weak引用指向的对象被释放时,又是如何去处理weak指针的呢?

1、调用objc_release

2、因为对象的引用计数为0,所以执行dealloc

3、在dealloc中,调用了_objc_rootDealloc函数

4、在_objc_rootDealloc中,调用了object_dispose函数

5、调用objc_destructInstance

6、最后调用objc_clear_deallocating,详细过程如下:

a. 从weak表中获取废弃对象的地址为键值的记录

b. 将包含在记录中的所有附有 weak修饰符变量的地址,赋值为 nil

c. 将weak表中该记录删除

d. 从引用计数表中删除废弃对象的地址为键值的记录

3、Swift mutating关键字的使用?

在Swift中,struct和enum是值类型 class是引用类型。

默认情况下,在结构体中,实例方法中是不可以修改值类型的属性,使用mutating后可修改属性的值。

struct Point {
    var x = 0
    var y = 0
    
    mutating func changePoint(x: Int, y: Int) {
        self.x += x
    }
    
}

在引用类型中(即class)中的方法默认情况下就可以修改属性值,不存在以上问题。

4、UIView和CALayer是什么关系?

从框架来说

CALayer 基于 QuartzCore 框架 UIView 基于 UIKit 框架

从父类来说

使用CALayer同样可以实现与UIImageView一样的展示效果。那么到底有什么区别呢? 其实,CALayer是直接继承自NSObject的,而UIView是直接继承自UIResponder的。 所以、相比于CALayer来说、UIView多了一个事件处理功能、也就是说、CALayer是不能处理用户的触摸事件的、而UIView可以。

那么、当我们展示出来的东西需要实现和用户交互的时候去使用UIView、而不需要的交互的时候CALayer和UIView都可以。

当然、由于CALayer不需要处理交互事件、所以是轻量级的、性能要比UIView高。

5、@synthesize 和 @dynamic 分别有什么作用?

@property有两个对应的词,一个是 @synthesize,一个是 @dynamic。如果 @synthesize和 @dynamic都没写,那么默认的就是@syntheszie var = _var;

@synthesize 的语义是如果你没有手动实现 setter 方法和 getter 方法,那么编译器会自动为你加上这两个方法 @dynamic 告诉编译器:属性的 setter 与 getter 方法由用户自己实现,不自动生成。