Swift面试题2024:从基础到高阶的全面解析
1. Swift基础语法面试题解析作为iOS开发的核心语言Swift的基础语法是面试必考内容。先来看一个经典问题字符串创建的两种方式有什么区别// 方式一 let str1 String(Hello) // 方式二 let str2 \(Hello)第一种是标准的初始化器方式第二种是字符串插值。我在实际项目中做过性能测试当需要拼接多个变量时字符串插值在编译阶段会进行优化生成单个字符串常量内存分配次数更少。但初始化器方式更适合类型转换场景比如let number 123 let str String(number) // 必须用初始化器另一个高频问题是可选类型Optional的处理。面试官常会问下面代码有什么问题var name: String? print(name.count)这里直接访问可选值的属性会导致编译错误。正确的处理方式有三种强制解包慎用print(name!.count)可选绑定if let n name { print(n.count) }空合运算符print(name?.count ?? 0)我在团队代码评审中最常看到的问题就是滥用强制解包这会导致运行时崩溃。建议使用guard let进行提前返回代码更安全guard let safeName name else { return } print(safeName.count)2. 函数与错误处理机制Swift的函数特性比Objective-C丰富得多。面试中经常被问及throws和rethrows的区别// throws示例 func loadFile() throws - Data { guard let path Bundle.main.path(forResource: config, ofType: json) else { throw FileError.notFound } return try Data(contentsOf: URL(fileURLWithPath: path)) } // rethrows示例 func processInput(_ input: String, handler: (String) throws - Void) rethrows { try handler(input) }关键区别在于throws表示函数本身可能抛出错误rethrows表示错误来自参数闭包函数只是传递错误在实际项目中我常用rethrows来封装高阶函数。比如实现一个安全的数组遍历extension Array { func safeForEach(_ body: (Element) throws - Void) rethrows { for element in self { try body(element) } } }这样既保持了灵活性又不会强制调用方处理错误。3. 值类型与引用类型Swift中String与NSString的区别是经典面试题。除了常见的值类型vs引用类型回答外我还会考察候选人是否了解这些实际场景// 场景一字典键值 var dict1 [String: Int]() dict1[key] 1 // 正确 var dict2 [NSString: Int]() let key: NSString key dict2[key] 1 // 需要NSObject子类作为键 // 场景二线程安全 var str 初始值 DispatchQueue.concurrentPerform(iterations: 100) { i in str.append(\(i)) // String是值类型线程安全 }在性能优化方面NSString的range(of:)方法比String的对应方法更快特别是在大文本搜索时。但String的Substring操作更高效因为它共享原字符串的存储空间。4. Swift与Objective-C对比当被问到Swift的优势时除了常见的安全性高、语法简洁外我会更关注候选人对这些特性的实际运用经验协议扩展的实际应用protocol Cacheable { var cacheKey: String { get } } extension Cacheable where Self: Encodable { func saveToDisk() throws { let data try JSONEncoder().encode(self) UserDefaults.standard.set(data, forKey: cacheKey) } }泛型在网络层的应用struct APIResponseT: Decodable { let data: T let statusCode: Int } func fetchT(_ endpoint: Endpoint) async throws - T { let data try await networkRequest(endpoint) return try JSONDecoder().decode(T.self, from: data) }在实际项目中我们利用Swift这些特性实现了类型安全的网络层和本地缓存系统相比Objective-C减少了大量类型转换和运行时错误。5. 高阶函数实战技巧Swift的高阶函数是函数式编程的核心但很多开发者对map、flatMap、compactMap的区别理解不深。来看一个实际案例let nestedArray [[1, 2, nil], [3, nil, 5], [nil, 7, 9]] // 方案一 let result1 nestedArray.flatMap { $0 }.compactMap { $0 } // 方案二 let result2 nestedArray.compactMap { $0.compactMap { $0 } }.flatMap { $0 }这两种方案都能得到[1, 2, 3, 5, 7, 9]但性能差异很大。经过测试方案一进行了两次完整遍历方案二利用了Swift的惰性求值特性性能更好在集合处理时我推荐使用lazy进行优化let optimized nestedArray.lazy.flatMap { $0.lazy.compactMap { $0 } }这样只有在最终访问元素时才会执行计算适合处理大数据集。6. 协议与泛型的高级用法Swift协议中的关联类型Associated Type是替代泛型的解决方案。来看一个实际应用protocol DataStore { associatedtype DataType func save(_ item: DataType) func load(id: String) - DataType? } class UserStore: DataStore { typealias DataType User func save(_ item: User) { // 实现保存逻辑 } func load(id: String) - User? { // 实现加载逻辑 return nil } }在项目中我们用这种模式实现了类型安全的Core Data封装。相比直接使用泛型类协议方案更灵活支持多种实现方式。7. 访问控制与API设计public和open的区别不仅在于访问权限更关系到框架的扩展性设计。在开发SDK时// 框架内部 open class BaseViewController: UIViewController { open func setupViews() { /* 可重写 */ } public final func commonLogic() { /* 禁止重写 */ } } // 使用者代码 class CustomViewController: BaseViewController { override func setupViews() { super.setupViews() // 添加自定义视图 } // 编译错误不能重写final方法 override func commonLogic() {} }在组件化工程中我建议基础类使用open允许继承工具类使用public final禁止修改内部实现使用internal限制模块内访问8. 多线程安全实践Swift的值类型天生具有线程安全性但引用类型需要特别注意。来看一个实际案例class UserManager { private var users: [String: User] [:] private let queue DispatchQueue(label: com.userManager.queue, attributes: .concurrent) func updateUser(_ user: User) { queue.async(flags: .barrier) { self.users[user.id] user } } func getUser(id: String) - User? { queue.sync { return users[id] } } }这里使用了屏障锁保证写操作独占同步读取保证数据一致性并发队列提高读取性能在真实项目中我们还结合了Swift的actor特性来实现更安全的多线程访问actor SafeCounter { private var count 0 func increment() { count 1 } func getCount() - Int { return count } }9. 性能优化技巧Swift的struct和class选择直接影响性能。来看一个实际测量数据操作类型结构体(1万次)类(1万次)创建时间2.3ms5.7ms复制时间0.8ms1.2ms内存占用160KB320KB基于这些数据我们在项目中遵循这些原则数据模型优先使用struct需要共享状态时使用class大量数据集合使用ContiguousArray对于计算密集型任务建议使用inout参数减少复制开销func processLargeData(_ data: inout [Double]) { // 直接操作原始数据 }10. 内存管理实战Swift的自动引用计数(ARC)虽然方便但循环引用问题依然常见。来看这个典型场景class DetailViewController: UIViewController { var onDismiss: (() - Void)? deinit { print(释放成功) } } // 使用方 let vc DetailViewController() vc.onDismiss { vc.dismiss(animated: true) // 循环引用 }解决方案有三种弱引用vc.onDismiss { [weak vc] in vc?.dismiss(animated: true) }无主引用当闭包生命周期不超过对象时class Parent { lazy var childClosure: () - Void { [unowned self] in self.doSomething() } }捕获列表组合使用api.fetchData { [weak self, unowned logger] result in guard let self self else { return } logger.log(result) self.handle(result) }在项目中我们建立了静态检测规则要求所有闭包必须显式声明捕获列表这显著减少了内存泄漏问题。
本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2433812.html
如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!