阅读视图

发现新文章,点击刷新页面。

Tauri 2 iOS 开发避坑指南:文件保存、Dialog 和 Documents 目录的那些坑

很多开发者在把 Tauri 2 应用上架到 iOS(真机或模拟器)时,都会在文件保存这一步踩坑:明明代码代码在其他平台没问题,在 iOS 路径就返回 null,或者在「文件」App 里根本看不到自己的 App 文件夹。

下面我把最常见的几个坑总结成一份避坑科普文,帮你一次性避开这些“iOS 特色”问题。

坑 1:@tauri-apps/plugin-dialogsave() 在 iOS 上经常返回 null 或路径不可用

现象
调用 const path = await save({...}) 后,一个 0KB 的文件写入成功,但是 path 返回是 null

避坑方法

  • 不要过度依赖 dialog.save() 来实现“用户任意选择保存位置”。
  • 优先使用 直接写入 App 的 Documents 目录(见坑 3)。
  • capabilities 中确保开启 dialog:save 权限。

坑 2:文件明明写入了,但「文件」App 里完全看不到 “Mind Elixir” 文件夹

现象: 用了 BaseDirectory.Document 保存文件后,在「文件」App → 浏览 → On My iPhone 里找不到你的 App 文件夹。

原因: iOS 沙盒机制严格控制 App 的 Documents 目录是否对「文件」App 可见。Tauri 默认生成的 iOS 项目不会自动添加暴露文件夹的配置,就算你写再多文件,文件夹也不会出现。

避坑方法(最关键的一步): 在 Info.plist 中添加以下两个 key(必须同时添加):

<key>UIFileSharingEnabled</key>
<true/>
<key>LSSupportsOpeningDocumentsInPlace</key>
<true/>

位置:通常在 src-tauri/gen/apple/ios/App/App/Info.plist(或你的项目对应路径),加在 <dict> 标签内,</dict> 之前。

添加后必须重新构建并安装 Appcargo tauri ios build 或用 Xcode 编译),然后:

  • 先执行一次写入操作(创建文件)。
  • 完全退出「文件」App(上滑关闭),重新打开并下拉刷新「On My iPhone」。

此时你应该能看到和 App 同名的文件夹(显示名称来自 productName 或 Xcode Display Name)。

注意:这两个 key 只控制可见性,不影响代码读写。

坑 3:iOS 上最好的保存方式其实不是 dialog,而是直接用 Documents 目录

推荐做法

import { writeTextFile, BaseDirectory } from '@tauri-apps/plugin-fs'

await writeTextFile('my-note.md', '你的内容...', {
  dir: BaseDirectory.Document
})

优点:

  • 最稳定,几乎不会出现 0 字节文件的问题。
  • 用户可以在「文件」App 里直接看到和管理文件(添加上面两个 plist key 后)。
  • 无需处理复杂的 URI 和潜在的 fs bug。

如果你想让用户输入文件名,可以结合 prompt 或自定义输入框实现。

总结建议

在 Tauri 2 + iOS 开发中:

  1. 优先使用 BaseDirectory.Document 直接保存(最稳)。
  2. 必须在 Info.plist 添加 UIFileSharingEnabledLSSupportsOpeningDocumentsInPlace
  3. 谨慎使用 dialog.save() + writeFile,因为移动端兼容性还有待完善(官方 issue 仍在跟进)。
  4. 开发时多用控制台日志 + Safari/XCode 调试,遇到路径问题先检查 plist 和权限。

避开这几个坑后,你的 Mind Elixir(或其他 App)在 iOS 上的文件保存功能就会顺畅很多。iOS 的沙盒和文件系统规则和桌面差异很大,提前了解这些“Apple 特色”能省下大量调试时间。

(本文基于 Tauri 2 常见 issue 和实际开发经验总结,iOS 规则可能随系统版本微调,建议以 Apple 官方文档为准。)

ViewModifier 和 圆角以及渐变色

ViewModifier

是什么

把一组样式或 UI 结构打包成可复用的东西,用 .modifier() 链式调用贴到任意 View 上。

类比 UIKit

UIKit 里你会封装一个函数来复用样式:

func styleToolButton(_ button: UIButton) {
    button.titleLabel?.font = .systemFont(ofSize: 25)
    button.setTitleColor(.white, for: .normal)
    button.frame.size = CGSize(width: 30, height: 30)
}

ViewModifier 干的是同一件事,但它不只能改属性,还能在原有 View 外面包一层新的 View 结构,这是普通函数做不到的:

struct BadgeModifier: ViewModifier {
    func body(content: Content) -> some View {
        ZStack(alignment: .topTrailing) {
            content  // 原来的 View 原封不动
            Text("99")
                .background(Color.red)
                .clipShape(Circle())
                .offset(x: 10, y: -10)
        }
    }
}

Image(systemName: "bell").modifier(BadgeModifier())
Image(systemName: "message").modifier(BadgeModifier())

本质

本质就是一个语法糖,功能上等价于自定义一个 View 然后把其他 View 塞进去,但它能融入 SwiftUI 的链式调用语法,用起来跟 .font() .foregroundColor() 一模一样。


圆角 + 渐变色 + 描边

是什么

SwiftUI 没有 UIKit 那样直接设置 layer.borderWidth 的属性,填充和描边需要两个图层叠加来实现。

类比 UIKit

UIKit 两行搞定:

view.layer.borderWidth = 4
view.layer.borderColor = UIColor.green.cgColor

SwiftUI 必须用 ZStack 叠两个 RoundedRectangle:

.background(
    ZStack {
        RoundedRectangle(cornerRadius: 20)
            .stroke(model.color, style: StrokeStyle(lineWidth: 4))
        RoundedRectangle(cornerRadius: 20)
            .fill(gradientStyle)
    }
)

为什么先 stroke 再 fill

stroke(描边)默认居中描边,线宽一半在内一半在外。fill 只填充内部区域,所以 fill 会覆盖 stroke 内侧的那一半。先画 stroke 再盖 fill,能让 stroke 外侧的一半露出来,边框视觉上更完整。反过来的话内侧边框线被盖住,边框显得细一半。

本质

这块 UIKit 确实更直观,SwiftUI 的声明式思路在这个场景下反而绕了一圈


本质

SwiftUI 没有 layer,只有 Shape + 绘制规则


fill 和 stroke 的区别

操作 本质
fill 填充 Shape 内部
stroke 沿路径画边

stroke 的问题

.stroke(lineWidth: 4)

👉 描边在路径两侧(内 + 外)


推荐方案(更精准)

.strokeBorder(lineWidth: 4)

👉 描边完全在内部


推荐结构

.fill(...)
.overlay(stroke)

👉 语义清晰:先填充,再叠加边框

ViewModifier

本质是 (View) -> View,不是修改 View,而是生成新 View


Modifier 顺序

顺序不是语法问题,而是 View 树结构

描边本质

边框不是属性,而是绘制结果(Shape + stroke)


Swift 6.3 正式发布支持 Android ,它能在跨平台发挥什么优势?

最近 Swift 发布了 6.3 版本,而这个版本最特殊的地方在于:把 Android SDK 作为首个官方发布版本给加了进来,其实这个话题在去年的 《Swift 官方正式支持 Android》我们就已经聊过,而这两天正式版的发布,也是广大 Swift 开发者最高涨的时刻,iOSer 终于也有了自己的原生跨平台基础了:

那 Swfit for Android 到底是什么?其实和之前我们聊的一样,目前并不是在 Android 上原生跑 SwiftUI ,这个能力目前是一个 SKIP 的第三方项目在做, Swfit for Android 主要完成了「Swift 官方支持把 Swift 代码交叉编译到 Android」。

所以 Swfit for Android 目前主要由三个部分组成:

  • Swift Toolchain:目标平台上的 Swift 编译器、标准库、LLVM 后端

  • Swift SDK for Android:给 Android 目标平台准备的 Swift 库、头文件、配置

  • Android NDK:提供 Android 的系统头文件、链接器、目标架构工具链等

也就是说,它的核心原理是「交叉编译」:

  • 需要 macOS / Linux 上装 Swift 工具链
  • 需要 Android target 的 Swift SDK artifact bundle
  • 需要 Android NDK 的 sysroot、linker、headers
  • 由 Swift 编译器把代码交叉编译成 Android 可执行文件或本地库(.so
swift build --swift-sdk x86_64-unknown-linux-android28 --static-swift-stdlib

最终结果来看,就是把代码的构建产物变成 Android 上可运行的 ELF 二进制。

其实这也是在 iOS 的 LLVM 的技术领域,比如 KMP 目前大多也是利用 iOS 分支的 LLVM 交叉编译还到鸿蒙 ,所以 Swift 编译器本身还是走在自己的前端和 LLVM 后端,只是 target 换成 Android,例如:

  • x86_64-unknown-linux-android28
  • aarch64-unknown-linux-android28

这里的 android28 可以看出来,是明确绑定到特定 Android API Level ,而 Swift runtime / Foundation 依赖的一些能力也需要较新的 Android API,所以会用 API 28 为基础。

当然,Swift 与 Android 的 Java/Kotlin 的协调桥梁不出意外是 JNI,当然这里不会让你手写 JNI,而是用自动桥接工具来完成。

而在实际应用层面,Swift 官方现在推荐的不是 “整 App 全 Swift”,而是“Swift 库 + Kotlin/Java 壳” ,比如官方 examples 仓库里推荐方案 `hello-swift-java,它的结构是:

  • 一个 Swift package / Swift library
  • 一个 Kotlin Android app(Jetpack Compose UI)
  • Kotlin 调 Swift,不需要你手写 JNI,交给 swift-java 自动生成 Java wrapper 和 JNI bindings

也就目前而言,推荐的是 business logic / algorithms / libraries 写成 Swift ,而前端仍然保持标准 Kotlin/Java Android app 形态 ,简单来说就是:

  • UI:Kotlin / Jetpack Compose
  • 共享逻辑:Swift
  • 桥接层:swift-java / JNI

这强烈的即视感,不就是最初的 KMP 那会么,CMP 还没支持 UI 的时候,KMP 也是这样的路线

比如 Swift 官方提供的例子:一个 Android ( weather-app )和 Swift 库( weather-lib ),用于获取当前位置的天气信息,weather-lib 内部使用 swift-openapi-generator 调用 OpenMeteo 天气 API ,并公开LocationFetcher protocol , 然后 swift-java 和 JNI 自动生成 Java Wrapper ,从而让 Kotlin 可以直接调用 Swift Library。

而在这次 Swift 官方开源的项目里,核心的项目就是:swift-javaswift-java-jni-core ,他们分别作为“上层桥接工具”与“底层 JNI 基建”支撑起整个 Swfit for Android 的生态。

其实也很有趣,Kotlin 在 Android 跑 JVM ,在 iOS 跑 KN 二进制;而现在反过来 Swift 跑 Android ,也是跑二进制。

swift-java-jni-core

swift-java-jni-core 是一个 Swift-friendly 的 JNI 低层封装,本质上是 jni.h 上的一层薄封装,加上一些预打包的类型转换能力,用来和 JVM / Android Runtime(ART)交互,也就是:

  • 负责 JVM/ART 句柄
  • 找类、找方法
  • 处理线程、锁、引用
  • 做 Java/Swift 类型桥接

它的整个结构大概为:

Sources/  
├── CSwiftJavaJNI/          // C 模块:纯 JNI 头文件 ABI  
└── SwiftJavaJNICore/       // Swift 模块:所有上层封装  
    ├── VirtualMachine/     // JVM 句柄、线程、锁  
    ├── BridgedValues/      // 类型桥接  
    └── *.swift             // 类型系统、签名、Mangling  

整个链路从 Swift 应用代码开始,通过 JavaValue 协议进行类型转换,然后通过 JavaVirtualMachine 管理 JVM 交互,最终利用 CSwiftJavaJNI 的 C 接口调用实际的 JVM 实现:

swift-java

swift-java 是更高一层的互操作工具平台,大致可以分为:

  • Swift 调 Java
  • Java 调 Swift
  • 自动生成绑定代码
  • Android 上支持 jextract --mode=jni,因为 Android/ART 没有服务端 Java 那套 FFM 路线的前提条件

在实现上主要有两个代码生成管道:

1、Swift 调用 Java

  • 通过反射分析 Java 类文件,生成 Swift Wrapper
  • 使用 Swift 宏(@JavaClass@JavaMethod)在编译时展开为 JNI 调用

更具体的就是通过 swift-java wrap-java 命令,工具在运行时利用 Java 反射读取 Java 类(包括 .jar 文件),给每个 Java 类生成对应的 Swift 类型,生成的 Swift 类型使用 @JavaClass@JavaMethod@JavaField 等宏进行标注

用途
@JavaClass 声明一个 Swift 类型是 Java 类的包装
@JavaInterface 声明一个 Swift 类型是 Java 接口的包装
@JavaMethod 将 Swift 方法变为调用 Java 方法的桥接
@JavaField 访问 Java 实例字段
@JavaStaticField 访问 Java 静态字段
@JavaImplementation 在 Swift 中实现 Java native 方法

例如,可以将HelloSwiftMain类型扩展为符合ParsableCommand接口,并使用 Swift 参数解析器来处理 Java 提供的参数:

import ArgumentParser
import SwiftJNI

@JavaClass("org.swift.jni.HelloSwiftMain")
struct HelloSwiftMain: ParsableCommand {
  @Option(name: .shortAndLong, help: "Enable verbose output")
  var verbose: Bool = false

  @JavaImplementation
  static func main(arguments: [String], environment: JNIEnvironment? = nil) {
    let command = Self.parseOrExit(arguments)
    command.run(environment: environment)
  }
  
  func run(environment: JNIEnvironment? = nil) {
    print("Verbose = \(verbose)")
  }
}

所以,对应也存在类型映射的需求:

Java type Swift type
boolean Bool
byte Int8
char UInt16
short Int16
int Int32
long Int64
float Float
double Double
void Void (rare)
T[] [T]
String String
Java class Swift class Swift module
java.lang.Object JavaObject SwiftJava
java.lang.Class<T> JavaClass<T> SwiftJava
java.lang.Throwable Throwable SwiftJava
java.net.URL URL JavaNet

2、Java 调用 Swift(jextract)

主要是让 Java 程序调用 Swift 库 ,生成 Java 绑定和 Swift thunk 文件,支持 FFM 和 JNI 两种生成模式。

比如 jextract 这个流程会分两步:

  • Swift Thunk:用 @_cdecl 暴露 C 符号入口
  • Java 绑定:生成 Java 代码,通过 JNI 或 FFM 调用这些 C 入口

所以,整个 swift-java 主要就是提供自动化方式生成 Swift/Java 绑定,其中:

  • jni 模式兼容性最广,主要支持 Android
  • FFM 模式更偏 Java 22+/25+ 服务端场景,不是 Android 主战场

也就是,swift-java 不只是支持 Android ,它还可以支持 Java Web 场景,野心还是有的。

而整个链路上其实是「Swift - 编译成本地库 - 通过生成的 JNI/Java wrapper 暴露给 Kotlin/Java 调用」 这样一个实现:

image-20260330104416812

那聊到这里,目前局限性也很明显了,它没有一个「 Swift UI for Android 」的支持,它能够让 iOS 的同学把自己的业务逻辑或者纯 Swift 代码共享给 Android ,但是 UI 还是得 Android 自己写。

另外,Swift 社区官方论坛里也有人提到:目前的 Swift Android SDK 下,二进制体积过大,其中一个原因是 Foundation 依赖的 ICU 很重,因为你需要把 Swift runtime、Foundation、ICU 这些东西都带到 Android app 。

同时,在系统 API 上还是差了点意思,比如你需要调用 Andorid 的系统 API 时,大概需要:

Swift ↔ JNI ↔ Java/Kotlin ↔ Android 系统 API

目前这个链路还没有全套完整的官方实现,大概需要后续社区和官方继续补齐,所以当前更多是逻辑和算法等场景的复用,直接调用系统 API 还是会麻烦一些

当然,最终使用过程里,也可以把 Swfit 打包成独立的 Lib,比如在官方例子里的 hello-swift-raw-jni-library ,通过就可以构建出 hello-swift-raw-jni-library-release.aar

./gradlew :hello-swift-raw-jni-library:bundleReleaseAar

所以对于 Swift for Android 来说,目前还是处于起步阶段,作为第一个正式版的起步。

最后,不得不说,语言的最终归宿就是跨平台,UI 的最终归宿也是,现在的 Swift for Android 就像是当年的 KMP ,从语言的跨平台开始切入,未来要发展, Swift UI for Android 的路径看起来也不是不可能,至于最终能否发展起来,这就考验 Swift 社区的运营水平了。

image.png

链接

github.com/swiftlang/s…

github.com/swiftlang/s…

www.swift.org/documentati…

一墙之隔,不同的时空 - 肘子的 Swift 周报 #129

一年一度的 Let's Vision 大会在上海如期举行,今年的主题是:“Born to Create, Powered by AI”。除了与 Swift、空间计算相关的常规 Session,大会还邀请了许多开发者分享他们在工作中对 AI 的应用与理解。通过这些讲师对 AI 工作流的介绍,我也受益匪浅。原本只能容纳 300 人的 AI 主题会场,里三层外三层站满了热情高涨的观众。

老司机 iOS 周报 #367 | 2026-03-30

ios-weekly
老司机 iOS 周报,只为你呈现有价值的信息。

你也可以为这个项目出一份力,如果发现有价值的信息、文章、工具等可以到 Issues 里提给我们,我们会尽快处理。记得写上推荐的理由哦。有建议和意见也欢迎到 Issues 提出。

文章

🐕 用好你的 jj - 重新思考 Agent 时代的版本控制

@阿权:文章是 onevcat(喵神)安利 jj(Jujutsu) 在 AI Agent 时代替代 Git 进行本地版本控制。jj 是可与 Git 无缝兼容的本地版本控制工具(兼容方式为远端仍是 Git 提交),核心优势在于以 change 为核心,无 staging area 等中间态,操作直观,完美适配 AI Agent “先生成后整理”的工作模式,无需开发者打断业务思考指挥版本操作,比 Git 更适合 Agent 参与的本地开发。

🐎 Copy-On-Write in Swift: Semantics, Misconceptions, and a Custom Implementation

@Barney:这篇文章把 Swift 中的 Copy-on-Write 讲得很清楚,重点不是重复 “值类型修改时会复制” 这类结论,而是澄清 COW 只是某些类型选择采用的实现策略,并不是所有 struct 天生自带的机制。作者先从“值类型外壳 + 引用类型存储”的经典结构切入,说明标准库集合为什么能同时兼顾值语义和复制成本;再结合 isKnownUniquelyReferenced(_:) 展示写入前如何判断底层存储是否需要分离,并用一个自定义 SharedBuffer 例子串起完整实现。后半部分还补充了 _read / _modify accessor 在减少额外复制上的作用,以及自定义 COW 真正值得引入的场景:数据量大、复制频繁、读多写少且又希望保留值语义。对需要设计高性能数据结构的同学,这是一篇兼顾原理和落地实现的好文章。

🐎 OpenMAIC

@JonyFang:OpenMAIC(Open Multi-Agent Interactive Classroom)是清华开源的 AI 互动课堂平台,能够将任意主题或文档一键转化为沉浸式学习体验。核心亮点包括:多智能体协作(AI 老师 + AI 同学实时授课讨论)、丰富场景类型(幻灯片、测验、HTML 交互模拟、项目制学习)、白板语音实时讲解,以及 OpenClaw 集成支持在飞书、Slack、Telegram 等 20+ 聊天应用中直接生成课堂。项目支持 Vercel 一键部署和 Docker 本地运行,兼容主流 LLM 服务商,开箱即用。

🐕 Array expression trailing closures in Swift

@Smallfly:这篇文章深入解析了 SE-0508 提案带来的 Swift 语法改进,解决了数组与字典类型长期存在的尾随闭包使用限制,让语言一致性与 API 设计灵活性得到显著提升。核心亮点包括:

历史痛点解决:此前 Swift 解析器因 [T][K:V] 的语法歧义,禁止在数组 / 字典类型表达式后使用尾随闭包,导致自定义初始化器(如 builder 风格、@resultBuilder API)必须使用 .initArray<T> 形式,破坏代码简洁性。SE-0508 移除该限制,允许 [String] { ... } 这类符合直觉的语法。

API 设计赋能:库作者现在可以为数组 / 字典设计更自然的 DSL 风格 API,比如基于 @resultBuilder 的集合初始化器、流式生成数组的构造函数,语法与自定义类型保持统一,降低开发者学习成本。

扩展交互能力:支持与 callAsFunction 特性结合,实现数组字面量后直接接闭包的转换操作(如 ["a","b"] { $0.uppercased() }),进一步提升代码表达力。

语言一致性提升:消除了集合类型与自定义类型在尾随闭包语法上的差异,让 Swift 语言的语法规则更统一,同时仅存在极窄的兼容性影响,整体是小而美的语法优化。

这个提案虽然没有引入新的 runtime 特性,但通过平滑语法边缘,为开发者带来更符合直觉的编码体验,尤其对依赖闭包初始化的集合 API 场景帮助显著。

🐎 Xcode 26 Compilation Cache

@david-clang:Xcode 26 Compilation Cache 的根本目标不仅是让编译器提速 5%,而是彻底停止重复已完成的工作。相比缺乏复用能力的 DerivedData,新机制在输入源未变时会直接提取缓存。这在切换分支、清理重建及高频 CI 场景下,能免去大量无谓的编译损耗。当然,若项目真正的瓶颈在于资源处理或繁杂的脚本,它也并非一劳永逸的银弹。

🐎 Testing with Event Streams

@AidenRao:这篇文章讲的是把一批“靠回调驱动的异步测试”从 XCTest 迁移到 Swift Testing 时,如何既验证回调是否发生,又验证发生顺序。作者对比了 XCTestExpectation、Swift Testing 的 confirmation(容易写成层层嵌套且难区分顺序),最终给出一个很实用的解法:用 AsyncStream 把回调事件“汇总成事件流”,再在测试里收集并断言事件序列,顺手还封装了一个小型 EventStream wrapper 来减少样板代码。

工具

App-Store-Connect-CLI

asc-cli 是一款强大的 App Store Connect 命令行工具。相比于 Fastlane 庞大的体系,它更加聚焦且现代。直接调用 Apple 官方的 App Store Connect API,提供简洁的命令来处理从 Beta 邀请到内购项创建的所有杂活。

最重要的是,它不需要你懂 Ruby,没有复杂的环境配置。 对于追求极致简洁、想在终端或 CI 环境中快速调动 App Store 能力的开发者来说,这是一款足以取代 Fastlane 大部分功能的利器。

代码

🐕 MotionEyes

@Cooper Chen:MotionEyes 是一个面向 AI Agent 的 SwiftUI 动画可观测性工具,它将原本“只能靠肉眼判断”的 UI 动画行为,转化为可量化、可分析的结构化数据。通过在应用中插入轻量级 tracing(如位置、几何、滚动等),开发者可以实时记录动画过程,并以时间序列日志形式还原真实运动轨迹。

项目的亮点在于其“ agent-first ”设计:不仅提供底层埋点能力,还配套自动化调试 workflow 和视觉分析工具,能够生成关键帧、网格标注、像素差异等结果,帮助精确定位动画异常。

相比传统调试方式,MotionEyes 更像一个“动画黑盒分析仪”,适用于排查错位、卡顿、时序错误等复杂 UI 问题。对于构建高质量交互动效或探索 AI 辅助开发流程的团队来说,这是一个非常有前瞻性的基础设施工具。

内推

重新开始更新「iOS 靠谱内推专题」,整理了最近明确在招人的岗位,供大家参考

具体信息请移步:https://www.yuque.com/iosalliance/article/bhutav 进行查看(如有招聘需求请联系 iTDriverr)

关注我们

我们是「老司机技术周报」,一个持续追求精品 iOS 内容的技术公众号,欢迎关注。

关注有礼,关注【老司机技术周报】,回复「2024」,领取 2024 及往年内参

同时也支持了 RSS 订阅:https://github.com/SwiftOldDriver/iOS-Weekly/releases.atom

说明

🚧 表示需某工具,🌟 表示编辑推荐

预计阅读时间:🐎 很快就能读完(1 - 10 mins);🐕 中等 (10 - 20 mins);🐢 慢(20+ mins)

Swift多线程方案-Concurrency

简介

Swift Concurrency(async/await)是从 Swift 5.5 开始引入的一套并发编程模型,用来替代传统的回调(callback)、闭包嵌套(callback hell)、以及部分 GCD 使用场景,让异步代码写起来像同步代码一样清晰。

async / await

  • async:标记函数是 异步函数
  • await:表示 等待异步结果

本质:await 会“挂起当前任务”,但不会阻塞线程

示例

func fetchUser() async -> String {
    return "Tom"
}

func loadData() async {
    let user = await fetchUser()
    print(user)
}

async throws

支持错误处理(替代 callback 的 error)

enum NetworkError: Error {
    case failed
}

func fetchData() async throws -> String {
    throw NetworkError.failed
}

func load() async {
    do {
        let result = try await fetchData()
        print(result)
    } catch {
        print("error: \(error)")
    }
}

Task

Task 是并发执行的基本单位,类似 GCD 的 block,也是一个对象。

  • 普通 Task
Task {
    let data = await fetchUser()
    print(data)
}
let task = Task {
    ...
}
  • Detached Task(独立线程)
Task.detached {
    await doSomething()
}

区别:

Task:继承当前 Actor / 优先级 / 上下文

detached:完全独立(慎用)

async let

并发执行

场景:多个接口同时请求

func loadData() async {
    async let user = fetchUser()
    async let posts = fetchPosts()
    
    let result = await (user, posts)
    print(result)
}

TaskGroup

任务组

场景:批量请求 / 并发处理列表

func fetchAll() async {
    // 每个子任务返回值类型是 String
    await withTaskGroup(of: String.self) { group in
        // 动态创建多个异步任务 并发执行,逐个获取结果
        for i in 1...3 {
            group.addTask {
                return "Task \(i)"
            }
        }
        
        for await result in group {
            print(result)  // 打印每个任务的结果;顺序不确定-先完成先打印
        }
    }
}

生命周期:当代码执行完withTaskGroup { ... },会自动等待所有子任务完成,然后自动释放资源。另外,也能自动取消未完成任务(如果提前退出)。任一任务抛错 就会全部取消。

手动取消 group.cancelAll() ,如 退出页面

使用场景:

  • 动态任务数量 (对比 async let - 任务数固定)
  • 适合列表/批量处理

1、批量接口请求

func fetchUsers(ids: [Int]) async -> [User] {
    var result: [User] = []
    
    await withTaskGroup(of: User.self) { group in
        for id in ids {
            group.addTask {
                return await fetchUser(id: id)
            }
        }
        
        for await user in group {
            result.append(user)
        }
    }
    
    return result
}

2、批量下载

func downloadImages(urls: [URL]) async -> [UIImage] {
    var images: [UIImage] = []
    
    await withTaskGroup(of: UIImage?.self) { group in
        for url in urls {
            group.addTask {
                return try? await downloadImage(url: url)
            }
        }
        
        for await img in group {
            if let img = img {
                images.append(img)
            }
        }
    }
    
    return images
}

3、并发计算任务(CPU)

func processData(items: [Int]) async {
    await withTaskGroup(of: Void.self) { group in
        for item in items {
            group.addTask {
                heavyWork(item)
            }
        }
    }
}

4、限制最大并发数(手动处理)

func fetchWithLimit(ids: [Int]) async {
    let maxConcurrent = 2
    await withTaskGroup(of: String.self) { group in
        var iterator = ids.makeIterator()
        // 先启动前 N 个任务
        for _ in 0..<maxConcurrent {
            if let id = iterator.next() {
                group.addTask {
                    return await fetchUser(id: id)
                }
            }
        }
        // 每完成一个,就补一个
        for await result in group {
            print(result)
            if let nextId = iterator.next() {
                group.addTask {
                    return await fetchUser(id: nextId)
                }
            }
        }
    }
}

Actor

线程安全方案

用于解决数据竞争问题(替代锁)

actor Counter {
    private var value = 0
    
    func increment() {
        value += 1
    }
    
    func getValue() -> Int {
        return value
    }
}
let counter = Counter()

Task {
    await counter.increment()
    let v = await counter.getValue()
    print(v)
}

Actor = 自动串行队列 + 数据隔离

对比

方案 问题
NSLock 易死锁
DispatchQueue 需要手动管理
Actor 天然安全

实战应用场景

  • 网络请求
func fetchUser() async -> User { ... }
func fetchPosts() async -> [Post] { ... }

func load() async {
    let user = await fetchUser()
    let posts = await fetchPosts()
}

——对比旧方案

fetchUser { result in
    fetchPosts { posts in
        // 嵌套地狱
    }
}
  • 多个接口并发请求

多个请求任务并行执行,等待异步结果

func loadPage() async {
    async let banner = fetchBanner()
    async let list = fetchList()
    async let profile = fetchProfile()
    
    let (b, l, p) = await (banner, list, profile)
}
  • 主线程更新UI
func loadData() {
    Task {
        let data = await fetchData()
        
        await MainActor.run {
            self.label.text = data
        }
    }
}

或 使用@MainActor

@MainActor
func updateUI() {
    label.text = "Hello"
}
  • 取消任务
let task = Task {
    let data = await fetchData()
}

task.cancel()
  • 图片加载
func downloadImage(url: URL) async throws -> UIImage {
    let (data, _) = try await URLSession.shared.data(from: url)
    return UIImage(data: data)!
}
let task = Task {
    let image = try await downloadImage(url: url)
    cell.imageView.image = image
}

注意:需要处理 cell 复用问题(取消任务)-》 避免图片错位

override func prepareForReuse() {
    task?.cancel()
}
  • 顺序依赖
func process() async {
    let token = await login()
    let data = await fetchData(token: token)
    let result = await parse(data)
}

对比

方案 特点
GCD 底层强,但难维护
Operation 可控但复杂
async/await 简洁 + 可读性强

本质: •GCD:你管理线程 •async/await:系统帮你调度

总结

Swift Concurrency 本质就是:

用“同步写法”写“异步代码”,并且保证线程安全

SwiftUI 如何实现 Infinite Scroll?

欢迎点个 star:github.com/RickeyBoy/R…

面试题:用 SwiftUI 实现一个无限滚动列表,支持分页加载。

这道题我在面试中遇到过好几次,说实话第一次答的时候以为随便写个 LazyVStack + onAppear 就完事了。后来才发现,面试官真正想考的不是你会不会用 API,而是你对状态管理、性能优化、Task 生命周期这些东西到底理解多深。

我的思路是从最简方案出发,一步步暴露问题、一步步优化。在开始写代码之前,先聊一下架构选型。

为什么选 MVVM?

先说一下 SwiftUI 里常见的架构选择。MVC 就不聊了,那是 UIKit 时代的标配,Controller 跟 UIKit 强耦合,到了 SwiftUI 里根本没有 UIViewController 这个角色,MVC 自然也就退出舞台了。

SwiftUI 里最常见的架构,从简单到复杂大概是这么几个:

架构 特点 适合场景
MV(Model-View) 没有 ViewModel,状态直接放 View 里,Apple 官方示例的典型写法 逻辑简单的页面
MVVM 抽出 ViewModel 管理状态和逻辑,SwiftUI 里最主流的选择 中等复杂度,需要可测试性
TCA 单向数据流,State + Action + Reducer + Effect,强约束 大型项目,需要严格的状态管理

其中 MV 是最基础的,逻辑简单的页面,@State 往 View 里一放就完事了,Apple 自己的 WWDC 示例大量都是这么写的。但 infinite scroll 涉及分页状态、加载状态、错误处理、Task 生命周期管理这些东西,全塞 View 里会很乱。抽一个 ViewModel 出来专门管理这些状态,View 只负责渲染和转发用户操作,职责就清晰多了。

所以这道题用 MVVM 是最合适的,不是因为 MVVM 最好,而是这个场景的复杂度刚好适合。并且采用 MVVM 结构规整,可拓展性也强,从面试回答的角度来讲也是正好的。

而 SwiftUI 天然就鼓励这种模式,@Observable 本身就是 binding 机制,ViewModel 状态一变,View 自动更新,不需要手动同步。我们后面的代码就是按这个思路来的。

一、最小可用版本

先写一个能跑的最简版本。

核心思路很简单:LazyVStack 只在 item 即将可见时才实例化 View,我们利用 onAppear 检测"最后一个 item 出现了",然后触发下一页请求。

Model

struct Item: Identifiable, Equatable {
    let id: String
    let title: String
}

struct PageInfo {
    let endCursor: String?
    let hasNextPage: Bool
}

ViewModel

@MainActor @Observable
final class ItemListViewModel {
    private(set) var items: [Item] = []
    private var pageInfo: PageInfo?

    func loadNextPage() async {
        let response = try? await APIService.fetchItems(after: pageInfo?.endCursor)
        guard let response else { return }
        items.append(contentsOf: response.items)
        pageInfo = response.pageInfo
    }
}

View

struct ItemListView: View {
    @State private var viewModel = ItemListViewModel()

    var body: some View {
        ScrollView {
            LazyVStack {
                ForEach(viewModel.items) { item in
                    ItemRow(item: item)
                        .onAppear {
                            if item == viewModel.items.last {
                                Task { await viewModel.loadNextPage() }
                            }
                        }
                }
            }
        }
        .task { await viewModel.loadNextPage() }
    }
}

代码很短,逻辑也直白:

  1. 每当最后一个 item 出现在屏幕上,就触发 loadNextPage()
  2. loadNextPage() 请求后台去 fetch,拿到数据然后塞进 items 中
  3. View 检测到有更新,自动刷新页面

一句话总结:最后一个 item onAppear 的时候,就进行请求。

1.1 分页方式:cursor vs offset

可能有同学会问:为什么 fetchItems(after: cursor) 用的是 cursor,而不是传统的 pageoffset

分页一般有两种方式:

  • Offset-basedfetchItems(page: 3, size: 20),按页码或偏移量取数据
  • Cursor-basedfetchItems(after: "abc123"),传上一页最后一条的标识,从那里往后取

对于 infinite scroll 这种场景,cursor-based 更合适。详细对比一下:

Cursor-based Offset-based
数据一致性 不受中间插入/删除影响 插入新数据会导致重复或遗漏
性能 数据库只需定位到 cursor 后续 大 offset 需要 skip N 行
适用场景 实时 feed、社交流 固定数据集、后台管理列表

简单来说,cursor-based 更适合"数据随时在变"的场景(比如社交 feed),offset-based 更适合"数据基本不变"的场景(比如后台管理列表)。infinite scroll 的数据通常是动态的,所以用 cursor-based。

1.2 LazyVStack vs List

可能有同学会问:为什么用 LazyVStack 而不是 List

先说浅显的回答:LazyVStack 布局更自由,没有 List 自带的分割线、背景色、cell 样式这些限制,适合高度自定义的 UI。而 List 开箱即用,自带滑动删除、拖拽排序这些交互,适合标准列表场景。

当然,如果想要深入回答,还有可以继续。二者还有一个关键区别其实是内存模型

LazyVStack List
View 回收 ❌ 不回收,创建后常驻内存 ✅ 内部回收机制
内存增长 随滚动距离线性增长 基本恒定
自定义布局 完全自由 受限于 List 样式
万级数据 可能有内存压力 表现更好

为什么会有这个区别?因为它们底层的实现不一样。List 底层是基于 UICollectionView(iOS 16 之前是 UITableView),天然有 cell 回收复用机制,滚出屏幕的 cell 会被回收,滚入时再复用,所以内存占用基本恒定。而 LazyVStack 底层只是一个普通的布局容器,"Lazy" 的意思是延迟创建,item 滚入可见区域时才创建 View,但创建之后就一直留在内存里,不会回收。

所以如果列表数据量很大(比如社交 feed 那种上万条的),List 在内存上更有优势。如果需要高度自定义的 UI,那就用 LazyVStack,但要心里有数:用户滚得越远,内存占用越大。

1.3 为什么加 @MainActor

上面的 ViewModel 代码加了 @MainActor,这个很容易被忽略但其实很关键。

@Observable 本身不会自动保证在主线程更新状态。而我们的 loadNextPage() 是在 Task 里通过 await 拿数据,await 之后的代码在哪个线程执行是不确定的。如果恰好在后台线程执行了 items.append(...),SwiftUI 收到状态变更通知后会在后台线程刷新 UI,这就会导致紫色警告("Publishing changes from background threads is not allowed")甚至崩溃。

加上 @MainActor 之后,这个类的所有属性访问和方法调用都会被隔离到主线程,从根源上避免线程安全问题。

另外补充一下:Swift 6.2(Xcode 26)引入了模块级别的 Default Actor Isolation 设置,可以把整个模块的默认隔离改为 MainActor,开启之后所有类型都默认跑在主线程,不用再手动加 @MainActor。但这是一个 opt-in 的设置,默认值还是 nonisolated,而且不是所有项目都会立刻升级。所以目前来说,显式写 @MainActor 仍然是更稳妥的做法。

1.4 几个小细节

有几个代码细节,不影响功能,但代码质量会好不少,属于面试加分项。

private(set) 控制可见性

itemsprivate(set) 修饰,外部只能读不能写。这样 View 就没法直接改 items,所有数据变更都必须经过 ViewModel 的方法,数据流向是单向的。这个习惯在 MVVM 里很重要,不然 View 和 ViewModel 的职责边界很容易模糊。

让 Item 遵循 Equatable

上面的代码里 Item 已经加了 Equatable,所以判断"是不是最后一个"可以直接写 item == viewModel.items.last,不用绕一圈去比 id。后面加叠加更多功能的时候也可以用,代码更简洁。

.task = .onAppear + Task

View 里首次加载用的是 .task { await viewModel.loadNextPage() },这其实等价于在 .onAppear 里手动创建一个 Task。但 .task 有个好处:当 View 消失时会自动 cancel 这个 Task。手动写 Task {} 的话你得自己管 cancel,容易漏掉,所以首次加载优先用 .task

二、防重复请求

上一个最基础的版本,有个明显的问题:快速滚动时 onAppear 有可能会被多次触发,导致同一页被重复请求了。

怎么解决?思路也很直接:加一个 isLoading 标记 + hasNextPage 判断,双重 guard,然后通过这些属性来判断是否需要发送请求。

@MainActor @Observable
final class ItemListViewModel {
    private(set) var items: [Item] = []
    private(set) var isLoading = false
    private var pageInfo: PageInfo?

    var canLoadMore: Bool {
        guard let pageInfo else { return items.isEmpty } // 首次加载
        return pageInfo.hasNextPage && !isLoading
    }

    func loadNextPage() async {
        guard canLoadMore else { return }
        isLoading = true
        defer { isLoading = false }

        let response = try? await APIService.fetchItems(after: pageInfo?.endCursor)
        guard let response else { return }
        items.append(contentsOf: response.items)
        pageInfo = response.pageInfo
    }
}

canLoadMore 这个 computed property 干了两件事:

  • 没有下一页时不请求(通过后端返回的 pageInfo.hasNextPage 来判断)
  • 正在加载时不重复请求(通过 isLoading 来判断)

2.1 小细节

defer 管理状态翻转

注意 isLoading 的写法:开头设为 true,然后紧接着 defer { isLoading = false }。这样不管后面是正常返回还是提前 returnisLoading 都会被重置回 false

如果不用 defer,你就得在每个 return 之前手动加一句 isLoading = false,路径一多很容易漏掉,漏掉的后果就是列表永远卡在 loading 状态,再也加载不了下一页。

canLoadMore 作为 computed property

把"能不能加载"的判断收到一个 computed property 里,而不是在 loadNextPage() 里写一堆 if。好处是逻辑集中,后面要加新条件(比如错误状态下不加载)直接改这一个地方就行,调用方不用动。

三、提前预加载:Threshold Prefetch

目前的逻辑是"最后一个 item 出现了才开始加载",那么用户的感受就是:滚到底 → 停顿 → 等数据 → 新数据出现。那个停顿虽然可能只有几百毫秒,但体感上还是挺明显的。

怎么办?提前触发。 不等最后一个 item,而是在还剩 N 个 item 时就开始加载下一页。

// View
ForEach(viewModel.items) { item in
    ItemRow(item: item)
        .onAppear { viewModel.onItemAppear(item) } // View 层仅透传,将逻辑交给 ViewModel
}

// ViewModel,新增 prefetch threshold
private let prefetchThreshold = 5

func onItemAppear(_ item: Item) {
    guard let index = items.firstIndex(of: item),
          index >= items.count - prefetchThreshold else { return } // 判断是否该加载下一页了
    Task { await loadNextPage() }
}

这样一来,用户还剩 5 个 item 可以滚的时候,网络请求就已经在跑了,等滚到底部时,数据大概率已经回来了,体验上就是"无缝衔接"。

那 threshold 到底设多少合适?这个纯属经验值,根据具体的数据量、UI 复杂度都相关,5 只是一个经验值。总的来讲就是一个 trade-off:

  • threshold 太小:快速滚动还是会看到停顿
  • threshold 太大:用户可能只看前几条就走了,白白浪费请求

四、Task 取消 + 错误处理

到这里基本功能已经没问题了。接下来聊聊 Task 生命周期管理和错误恢复,这部分在面试里属于加分项。

4.1 Task 取消

为什么需要管理 Task 取消?我们目前的例子中,单一列表的情况可能不需要考虑。但是如果是搜索页面的列表,或者叠加筛选功能,问题就复杂了。

举个具体的例子:

  1. 用户在搜索页搜"咖啡",然后在列表页向下滑动,触发了一个 loadNextPage 的请求 A
  2. 还没等数据回来,用户改成搜"奶茶",请求 B 又发出去了
  3. 这时候网络上同时有两个请求在跑。如果请求 B 先于 A 回来,那么等请求 A 回来的时候,用户就会发现明明搜索的是“奶茶”,但是却又展示了不少“咖啡”内容。

这种 bug 不是每次都能复现(取决于网络时序),但一旦出现用户会很困惑,所以解决方式就是:发新请求前先 cancel 旧的,被 cancel 的任务即便返回了 response 也不处理

@MainActor @Observable
final class ItemListViewModel {
    private(set) var items: [Item] = []
    private(set) var isLoading = false
    private(set) var error: Error?
    private var pageInfo: PageInfo?
    private var loadTask: Task<Void, Never>? // 💾 持有当前请求的引用

    func loadNextPage() {
        guard canLoadMore else { return }
        loadTask?.cancel() // ❌ 发新请求前,先 cancel 旧的
        isLoading = true

        loadTask = Task { [weak self] in // 🔒 weak self 防止循环引用
            guard let self else { return }
            defer { self.isLoading = false }

            do {
                let response = try await APIService.fetchItems(after: pageInfo?.endCursor)
                guard !Task.isCancelled else { return } // 🛡️ 被 cancel 了就不写入
                self.items.append(contentsOf: response.items)
                self.pageInfo = response.pageInfo
                self.error = nil
            } catch {
                guard !Task.isCancelled else { return } // 🛡️ 同上
                self.error = error
            }
        }
    }

    func reset() {
        loadTask?.cancel() // ❌ 先 cancel,再清空
        items = []
        pageInfo = nil
        isLoading = false
        error = nil
    }

    // ...
}

4.2 错误重试

错误处理其实是一个很容易被忽略,同时也非常复杂的事情。这里我们的方案是当出现错误的时候,展现一个重试按钮。从 UI 的角度来讲不好看,但实际上面试阶段时间有限,能够展示出有错误处理的思维就可以了。

@MainActor @Observable
final class ItemListViewModel {
    // ...
    private(set) var error: Error?

    func retry() {
        error = nil
        loadNextPage()
    }
}
// View — 列表底部
if viewModel.error != nil {
    RetryButton { viewModel.retry() }
} else if viewModel.isLoading {
    ProgressView()
}

4.3 空状态处理

还有一个容易忽略的边界情况:首次加载完成后,后端返回了 0 条数据。

当前的代码里,items 为空有两种可能:一种是"还在加载第一页",另一种是"加载完了但确实没数据"。如果不区分这两种状态,用户看到的就是一片空白,不知道是在等数据还是真的没有内容。

处理方式也很简单,加一个 computed property 判断一下:

var isEmpty: Bool {
    !isLoading && items.isEmpty && error == nil && pageInfo != nil
}

这里的关键是 pageInfo != nil,说明至少请求过一次了(首次加载前 pageInfonil)。四个条件同时满足,才说明"确实没数据"。

View 里对应的处理:

if viewModel.isEmpty {
    ContentUnavailableView("暂无数据", systemImage: "tray")
} else if viewModel.isLoading && viewModel.items.isEmpty {
    ProgressView() // 首次加载中
} else {
    // 正常的列表内容
}

这样用户就能清楚地区分"加载中"和"没有数据"这两种状态了。

4.4 用 enum 收敛 View 状态

到这里你会发现,View 层需要处理的状态越来越多:首次加载中、有数据、空数据、出错。如果全用 if/else if 判断,条件一多很容易写乱,漏掉某个分支也不会有编译器提醒。

可以定义一个 enum 来收敛这些状态:

enum ViewState {
    case initialLoading    // 首次加载中
    case loaded            // 有数据,正常展示列表
    case empty             // 加载完了但没数据
    case error(String)     // 出错了
}

然后在 ViewModel 里加一个 computed property,从现有属性推导出当前的 View 状态:

var viewState: ViewState {
    if let error, items.isEmpty {
        return .error(error.localizedDescription)
    }
    if isLoading && items.isEmpty {
        return .initialLoading
    }
    if isEmpty {
        return .empty
    }
    return .loaded
}

注意这里的关键:ViewState 是 computed property,不是存储属性。底层的数据源还是 isLoadingitemserrorpageInfo 这些独立属性,viewState 只是把它们组合成 View 更容易消费的形式。这样既不会出现之前 LoadingState enum 耦合状态的问题,又让 View 的代码变得很干净:

var body: some View {
    Group {
        switch viewModel.viewState {
        case .initialLoading:
            ProgressView()
        case .empty:
            ContentUnavailableView("暂无数据", systemImage: "tray")
        case .error(let message):
            ErrorView(message: message) { viewModel.retry() }
        case .loaded:
            ScrollView {
                LazyVStack(spacing: 0) {
                    ForEach(viewModel.items) { item in
                        ItemRow(item: item)
                            .onAppear { viewModel.onItemAppear(item) }
                    }
                    loadingFooter
                }
            }
        }
    }
    .task { viewModel.loadNextPage() }
}

switch 替代 if/else if,每个分支对应一种状态,漏掉任何一个编译器都会报错。用 Group 包裹 switch 是为了能在外层挂 .task 触发首次加载。

五、完整代码

前面一步步拆解完了,最后把所有东西整合到一起。先看一下整体架构:

graph LR
    View -->|用户操作| ViewModel
    ViewModel -->|状态更新| View
    ViewModel -->|网络请求| APIService
    APIService -->|响应数据| ViewModel

    style View fill:#E8F5E9,stroke:#4CAF50
    style ViewModel fill:#E3F2FD,stroke:#2196F3
    style APIService fill:#FFF3E0,stroke:#FF9800

View 只管渲染和转发用户操作,ViewModel 管状态和请求编排,APIService 做实际的网络调用。数据流向是单向的:用户操作 → ViewModel 处理 → APIService 请求 → 数据回来更新状态 → View 自动刷新。

Model

struct Item: Identifiable, Equatable {
    let id: String
    let title: String
}

struct PageInfo: Equatable {
    let endCursor: String?
    let hasNextPage: Bool
}

struct PagedResponse {
    let items: [Item]
    let pageInfo: PageInfo
}

ViewState

enum ViewState {
    case initialLoading
    case loaded
    case empty
    case error(String)
}

ViewModel

@MainActor @Observable
final class ItemListViewModel {
    // MARK: - State

    private(set) var items: [Item] = []
    private(set) var isLoading = false
    private(set) var error: Error?

    // MARK: - Private

    private let prefetchThreshold = 5
    private var pageInfo: PageInfo?
    private var loadTask: Task<Void, Never>?

    // MARK: - Computed

    var canLoadMore: Bool {
        guard !isLoading else { return false }
        guard let pageInfo else { return items.isEmpty }
        return pageInfo.hasNextPage
    }

    var isEmpty: Bool {
        !isLoading && items.isEmpty && error == nil && pageInfo != nil
    }

    var viewState: ViewState {
        if let error, items.isEmpty {
            return .error(error.localizedDescription)
        }
        if isLoading && items.isEmpty {
            return .initialLoading
        }
        if isEmpty {
            return .empty
        }
        return .loaded
    }

    // MARK: - Trigger

    func onItemAppear(_ item: Item) {
        guard let index = items.firstIndex(of: item),
              index >= items.count - prefetchThreshold else { return }
        loadNextPage()
    }

    // MARK: - Actions

    func loadNextPage() {
        guard canLoadMore else { return }
        loadTask?.cancel()
        isLoading = true

        loadTask = Task { [weak self] in
            guard let self else { return }
            defer { self.isLoading = false }

            do {
                let response = try await APIService.fetchItems(after: pageInfo?.endCursor)
                guard !Task.isCancelled else { return }
                self.items.append(contentsOf: response.items)
                self.pageInfo = response.pageInfo
                self.error = nil
            } catch is CancellationError {
                // Task was cancelled, do nothing
            } catch {
                guard !Task.isCancelled else { return }
                self.error = error
            }
        }
    }

    func retry() {
        error = nil
        loadNextPage()
    }

    func reset() {
        loadTask?.cancel()
        items = []
        pageInfo = nil
        isLoading = false
        error = nil
    }
}

View

struct ItemListView: View {
    @State private var viewModel = ItemListViewModel()

    var body: some View {
        Group {
            switch viewModel.viewState {
            case .initialLoading:
                ProgressView()
            case .empty:
                ContentUnavailableView("暂无数据", systemImage: "tray")
            case .error(let message):
                ErrorView(message: message) { viewModel.retry() }
            case .loaded:
                ScrollView {
                    LazyVStack(spacing: 0) {
                        ForEach(viewModel.items) { item in
                            ItemRow(item: item)
                                .onAppear { viewModel.onItemAppear(item) }
                        }
                        loadingFooter
                    }
                }
            }
        }
        .task { viewModel.loadNextPage() }
    }

    @ViewBuilder
    private var loadingFooter: some View {
        if viewModel.error != nil {
            VStack(spacing: 8) {
                Text("加载失败")
                    .font(.caption)
                    .foregroundStyle(.secondary)
                Button("Retry") { viewModel.retry() }
                    .buttonStyle(.bordered)
            }
            .frame(maxWidth: .infinity)
            .padding()
        } else if viewModel.isLoading {
            ProgressView()
                .frame(maxWidth: .infinity)
                .padding()
        }
    }
}

总结

回顾一下整个思路:

  1. 从简单方案说起LazyVStack + onAppear last item,先把原理讲清楚
  2. 暴露问题并优化 — 重复请求 → guard;体验停顿 → threshold prefetch
  3. 展示工程素养 — Task 取消、error handling、retry
  4. 完整架构 — View 只渲染 + 转发,ViewModel 管状态 + 编排

厌倦了那些看着像一个模版复刻出来的抓包工具,我开发了一款iOS端HTTPS抓包调试工具

最近的一份工作,因为对业务不熟悉,产品经理出的需求又不考虑历史兼容性,问同事同事也不清楚,作为一个后端开发,我也拿不到客户端的代码,于是我就想到了抓包,通过安装app,抓取某块功能使用了哪些接口。

因为我手机是iPhone, 我因此试用了很多款在app store下载的HTTPS抓包工具,包括免费的Stream、ProxyPin、付费了一款螃蟹抓包。但这些工具感觉都是出自于同一个模版,体验雷同,因为没有别得选择,当时只好忍受。

当时没被满足的一些需求:

1、发现一些图片无法抓取到(我想知道图片用的域名和路径,知道是直接访问云存储,还是用的哪个文件系统服务,这在后端项目中看不出来,因为这个项目的后端也没提供文件上传功能)。

2、JSON无高亮、无搜索功能,也无法对比某个业务参数(比如当商品类型是电子钥匙时、以及商品类型是摄像头时,实际传的参数以及响应的Body有哪些不同的字段)。

3、除体验外,我当时还希望能满足我这个需求:我想把这些接口导入到Apifox,并且基于当前接口和新的迭代需求在此基础上去修改接口,并在团队中共享这份接口。 而当时我只能基于抓取的响应结构,自己在Apifox里面写接口,这耗费了我整整一天时间。

经过那次之后,我决定自己研究写一个,这个HTTPS抓包工具一定把用户体验做好,一定支持抓图片、支持JSON高亮和搜索(甚至是JSON Diff),以及支持自动生成API文档,可以一键导出到Apifox。

2026年1月我开发出来了,这款APP就叫ApiCatcher(因为一开始的目的就是抓API的,所以取名ApiCatcher),所有产品功能皆为原创设计。

能做出来要感谢那些开源项目的,比如ProxyPin,或许是因为开源项目没有盈利,所以体验没做好吧。我似乎也能理解为什么大多数抓包工具长得那么相似了。

我研究了他们的核心抓包功能是如何实现,用了哪些技术,然后自己花两周时间在Claude辅助下用Swift造了一份轮子(就是核心的NIO代理服务器以及SSL握手),在此基础又花两周时间做了优化性能,降低CPU和内存的占用,同时支持抓取大文件请求,避免进程被系统kill掉。我使用SwiftData和文件来存储抓包数据,将请求和响应Body存文件,其它字符串存SwiftData,然后通过边读边写文件来降低对内存的占用,而SwitData则提供更强大的搜索能力,这为产品做查询过滤功能提供了支持,所以ApiCatcher支持非常多的过滤条件。

以下是产品最初几个核心功能的产品设计:

1、极简风格的抓包页面。(我还加了个小创意:正在抓包中的背景是一张蜘蛛网,有一只蜘蛛在上面爬) ApiCatcher | HTTPS抓包工具

2、请求详情内容聚合,便于在手机这种小设备上更好的查看数据,同时减少操作步骤。请求响应的每个部分都是一个卡片,卡片可展开收起。Body可导出和一键复制。Body可展开全屏预览。Body目前支持渲染图片、svg、html、xml和json。 ApiCatcher |请求详情页

3、JSON格式化、高亮、搜索、Diff支持: ApiCatcher | JSON格式化、高亮、搜索、Diff支持

4、接口文档自动生成,以及导出接口文档到Apifox等API调试工具,因为海外用户不用Apifox,所以也支持了Postman和Bruno: ApiCatcher | api导出到Apifox、Postman、Bruno

5、可以抓文件,其实任何HTTP请求都支持,不仅仅是图片,而且没有限制图片大小,多大都能抓,这些图片还可以导出来拿来测试用(一些需要上传特定图片测试的接口):

在这里插入图片描述

经过两个月时间,加上有不少用户给我提需求,于是慢慢功能都完善了。基本app store上的https抓包工具有的功能ApiCatcher都支持了,并且体验更好,像一些正则表达式、脚本都集成AI生成功能提升效率,让用户自己填API Key 。

工具本就是为开发者提升工作效率而开发,所以我们做了支持导入企业内部使用的受信的自签私钥和证书,也可以自己开发一个接收器实时接收抓包流量,实现API扫描分析需求。

这款工具不支持iOS17以下系统,因为用了SwiftData,SwiftData需要17.0以上才支持。整个项目纯SwiftUI开发,核心功能代码用swift-nio等apple官网库。代码高亮则用了WebView+CodeMirror+Highlight.js以及一些插件。这些在app关于我们->开源组件许可都有声明。

ApiCatcherChatTCP这两款网络数据包抓包分析工具都是我自己原创设计、开发的作品,目前两款产品在海外还是不少用户喜欢的,我知道国内大家都喜欢用免费的,比如Stream、ProxyPin、Reqable,但我还是要在各个平台上分享一下的,避免后面被人借鉴反被别人说是我们抄袭,赚不赚钱是次要的,得先证明自己是原创的。

不同阶段的 iOS 应用混淆工具怎么组合使用,源码混淆、IPA混淆

如果把 iOS 应用的混淆只理解成改类名,就会低估这个问题。实际项目里,信息暴露点分散在多个阶段,源码命名、编译产物、资源目录、甚至签名后的 IPA 结构。只用一个工具,很难覆盖完整路径。

这篇文章沿着构建流程往下走,看看每个阶段可以做什么处理,以及不同工具如何拼在一起使用。

在源码阶段先做可控改名

项目还在开发阶段时,可以先处理一部分明显暴露语义的命名,例如:

class VipSubscriptionManager
class PaymentOrderController

如果直接进入编译阶段,这些名称会被带入二进制。

可以通过脚本做一轮批量替换,例如:

  • 使用 Python 脚本扫描类名
  • 生成映射表
  • 替换为无语义名称

这一步的特点是:

  • 控制粒度高
  • 需要改动工程
  • 对团队规范有要求

如果项目已经稳定,这一步不一定适合继续做。

利用 Xcode 构建参数裁剪符号

进入构建阶段,可以先减少一部分信息暴露。

在 Release 配置中:

Strip Debug Symbols = YES
Dead Code Stripping = YES

构建后检查:

strings AppBinary | head

输出会比 Debug 包干净,但核心类名仍然存在。

这一阶段主要是“减少冗余”,不是混淆。

用命令行工具检查当前暴露程度

在进入下一步之前,可以用工具做一次快速判断:

strings AppBinary | grep ViewController

如果输出类似:

LoginViewController
ProfileViewController

说明结构仍然清晰,也可以用:

  • class-dump 查看接口
  • Hopper 查看符号表

这一步的目的是明确需要处理的范围。


在 IPA 层做统一混淆

当项目已经打包成 IPA 后,可以用专门的 iOS 应用混淆工具进行处理。

这里引入 Ipa Guard,它的处理方式不是修改源码,而是直接解析 Mach-O 文件并替换符号。

操作流程:

  1. 打开工具,加载 IPA
  2. 进入代码模块
  3. 选择需要处理的内容

可以看到:

OC 类
Swift 类
OC 方法
Swift 方法

代码混淆

在实际项目中,我们会筛选:

UserManager
PaymentService
VipController

执行混淆后:

UserManager → a82k3

再次用 strings 查看,原名称不会再出现。


资源文件处理不要忽略

很多人只处理代码,但资源同样是入口。

例如:

config/payment.json
assets/vip_banner.png

这些文件名称直接说明业务。

Ipa Guard 的资源模块可以:

  • 批量改名
  • 更新引用路径

处理后:

payment.json → x92ks.json
vip_banner.png → a8d3k.png

重命名


引入前端工具处理 JS / H5

如果项目中有 WebView 或 H5 页面,仅改名不够。

可以在构建阶段执行:

terser main.js -o main.min.js

或:

uglifyjs page.js -o page.min.js

压缩后再交给 IPA 混淆工具处理文件名。

这样组合后:

  • 内容不可读
  • 文件名无语义

修改资源指纹用于打散特征

当多个应用使用相同资源时,文件内容会成为识别依据。

Ipa Guard 支持修改资源 MD5:

md5 banner.png

处理前后结果不同。

这一层不影响功能,但会改变资源特征。 md5


清理调试信息

很多项目在 Release 包中仍然保留日志。

可以检查:

strings AppBinary | grep NSLog

如果输出较多,可以在 IPA 处理阶段删除。

Ipa Guard 支持清理调试信息,使二进制更简洁。


签名工具补上最后一步

所有修改完成后,必须重新签名。

可以使用:

kxsign sign app.ipa \
-c cert.p12 \
-p password \
-m dev.mobileprovision \
-z test.ipa \
-i

或者直接在 Ipa Guard 中配置签名参数。

安装到设备后,验证:

  • 页面是否正常
  • 动态调用是否有效
  • 资源是否加载

重签名


iOS 应用混淆不是某个工具的功能,而是一整条流程。源码阶段、构建阶段、IPA 阶段,各自能做的事情不同。把这些步骤串起来,比单独使用某一个工具更有效。

参考链接:ipaguard.com/blog/161

关于Xcode26.4 踩坑适配

Xcode26.4 踩坑适配

不建议升级Xcode 26.4,Xcode底部控制台无法使用po命令;

iOS 26.4模拟器启动加载巨缓慢,建议保持26.3.1。

随着 Xcode 26.4 正式版发布,编译器对私有头文件访问链式比较语法C++标准库特化的校验规则进一步收紧,导致 iOS 开发中常用的 AFNetworking、YYText、WCDB 三个主流第三方库出现编译报错/警告。本文针对这三类问题提供修复方案,帮助开发者快速完成 Xcode 26.4 适配。

一、AFNetworking:私有头文件访问报错

报错信息

Use of private header from outside its module: 'netinet6/in6.h'

问题原因

Xcode 26.4 强化了模块私有头文件的访问权限校验,AFNetworking 源码中直接引入了系统私有头文件 <netinet6/in6.h>,违反了 Xcode 的模块访问规则,触发编译报错。

解决方案

直接注释掉AFNetworking 中引入该私有头的代码行,无需其他修改即可解决。

  1. 找到 AFNetworking 中包含 #import <netinet6/in6.h> 的文件(通常为AFURLSessionManager.m或核心头文件);
  2. 注释该行代码:
// #import <netinet6/in6.h>
  1. Clean 项目缓存,重新编译即可。

二、YYText:链式比较语法错误

报错信息

Chained comparison 'X < Y < Z' does not behave the same as a mathematical expression

问题原因

Xcode 26.4 编译器对链式比较语法做了严格校验:X < Y < Z 在 OC/C 语言中并非数学意义的连续比较,而是先计算X<Y得到布尔值(0/1),再用该值与 Z 比较,逻辑完全错误。编译器会强制抛出警告,影响编译流程。

解决方案

前半段比较逻辑添加括号,明确运算优先级,修复语法歧义。

代码修改
[self _insideComposedCharacterSequences:line position:position block: ^(CGFloat left, CGFloat right, NSUInteger prev, NSUInteger next) {
    if (isVertical) {
-        position = fabs(left - point.y) < fabs(right - point.y) < (right ? prev : next);
+        position = (fabs(left - point.y) < fabs(right - point.y)) < (right ? prev : next);
    } else {
-        position = fabs(left - point.x) < fabs(right - point.x) < (right ? prev : next);
+        position = (fabs(left - point.x) < fabs(right - point.x)) < (right ? prev : next);
    }
}];
  1. 按上述代码添加括号;
  2. 重新编译,错误自动消失。

三、WCDB:C++标准库特化报错

报错信息

'is_integral' cannot be specialized: Users are not allowed to specialize this standard library entity

报错文件:Tag.hpp

报错截图

问题原因

Xcode 26.4 升级了底层 Clang/LLVM 编译器,严格遵循 C++标准规范:禁止开发者手动特化std::is_integral等标准库实体,WCDB 旧版源码的 Tag.hpp 文件触发了该规则限制。

解决方案

官方暂未提供修复方案,推荐两种任选其一

方案 1:其他开发者提交的修复 PR(源码修改)

直接应用 WCDB 其他开发者针对该问题的修复 PR,一键修复源码:

  • PR 地址:#1540
  • 操作:拉取 PR 代码替换本地 Tag.hpp 文件,重新编译即可。
方案 2:脚本打包 XCFramework(推荐)

使用 WCDB 官方脚本打包为xcframework,绕过源码编译的规则限制:

  1. 进入 WCDB 源码根目录;
  2. 执行官方打包脚本:
# 路径:/tools/version/build_xcframework.sh

./build_xcframework.sh \
  --scheme WCDBObjc \
  --configuration Release \
  --platforms ios ios-simulator \
  --output ./wcdb_xcframework
🧩 Creating XCFramework for WCDBObjc ...

[cmd] xcodebuild -create-xcframework -archive /Users/fjl/GitHub/wcdb/./wcdb_xcframework/archives/WCDBObjc-ios.xcarchive -framework WCDBObjc.framework -archive /Users/fjl/GitHub/wcdb/./wcdb_xcframework/archives/WCDBObjc-ios-simulator.xcarchive -framework WCDBObjc.framework -output /Users/fjl/GitHub/wcdb/./wcdb_xcframework/xcframeworks/WCDBObjc.xcframework

xcframework successfully written out to: /Users/fjl/GitHub/wcdb/wcdb_xcframework/xcframeworks/WCDBObjc.xcframework

✅ Created XCFramework: /Users/fjl/GitHub/wcdb/./wcdb_xcframework/xcframeworks/WCDBObjc.xcframework

⏱ Total elapsed time: 1 min 7 sec (67 s)
  1. 用生成的 xcframework 替换项目中原有 WCDB 的集成方式;
  2. Clean 项目后编译,问题彻底解决。

参考链接:WCDB Tag.hpp 报错官方 issue


适配总结

  1. AFNetworking:注释私有头引入行,解决模块访问权限问题;
  2. YYText:链式比较加括号,修复编译器语法校验;
  3. WCDB:合并官方 PR 或脚本打包 xcframework,解决 C++标准库特化限制。

完成以上修改后,清理 Xcode 缓存(Cmd+Shift+K),即可适配 Xcode 26.4,正常编译运行。

总结

  1. 三个第三方库的报错均由 Xcode 26.4编译器规则升级导致,修复无需改动业务代码;
  2. AFNetworking、YYText 为轻量代码修改,WCDB 推荐用官方脚本打包方案,稳定性更高;
  3. 适配后务必清理项目缓存,避免编译缓存残留问题。

iOS 26 适配 | 使用 `hidesSharedBackground` 保持导航栏按钮原有样式

iOS 26 适配 | 使用 hidesSharedBackground 保持导航栏按钮原有样式

背景

iOS 26 引入了全新的液态玻璃(Liquid Glass)设计语言,导航栏按钮的默认视觉风格发生了较大变化——多个按钮会被合并在一个统一的玻璃背景块中展示。对于希望在 iOS 26 下保持 iOS 26 之前导航栏按钮样式的开发者来说,苹果提供了 hidesSharedBackground API,用于将共享背景拆分,让每个 item 拥有独立的 Liquid Glass 背景:

if (@available(iOS 26.0, *)) {
    item.hidesSharedBackground = YES;
}

启用后,每个 item 的玻璃背景块会被单独渲染,视觉上更接近旧版导航栏中按钮各自独立的呈现方式。但问题随之而来:系统会在每个玻璃背景块之间插入默认间距,开发者无法通过常规 API 将这个间距收紧为 0,导致多个按钮之间出现明显的视觉割裂感,与 iOS 26 之前的紧凑排列效果存在差异。

因此,仅设置 hidesSharedBackground = YES 还不够,还需要额外处理 PlatterView 的间距问题,才能真正还原旧版导航栏的按钮布局样式。


问题根因分析

在 iOS 26 中,每个 UIBarButtonItem 的 Liquid Glass 背景块由私有容器 _UINavigationBarPlatterView 承载。

UINavigationBar
  └── _UINavigationBarContentView
        ├── _UINavigationBarPlatterView   ← 左侧按钮容器(含独立玻璃背景)
        │     └── _UIButtonBarButton
        └── _UINavigationBarPlatterView   ← 右侧按钮容器(含独立玻璃背景)
              └── _UIButtonBarButton

每个 PlatterView 负责绘制该按钮的 Liquid Glass 背景块,同时也决定了按钮在导航栏中的排列位置。系统在计算这些容器的布局时,会在相邻 PlatterView 之间注入固定的默认间距,且这个间距:

  • 无法通过 UIBarButtonSystemItemFixedSpace 负间距消除(iOS 26 已失效)
  • 无法通过修改 customView 的约束影响
  • 无法通过 UINavigationBar 的公开布局 API 干预

解决方案

核心思路:在布局完成后,运行时递归查找所有 PlatterView 容器,强制重置其 x 坐标与 Leading 约束,将相邻玻璃背景块之间的间距收紧为 0,从而还原 iOS 26 之前导航栏按钮的紧凑排列效果。

完整代码

#pragma mark - iOS 26 PlatterView 间距修复

- (void)fixPlatterViewSpace {
    // 收集所有 PlatterView
    NSMutableArray<UIView *> *platterViews = [NSMutableArray array];
    [self collectPlatterViews:self result:platterViews];
    
    if (platterViews.count == 0) return;
    
    CGFloat navBarWidth = self.frame.size.width;
    CGFloat midX = navBarWidth / 2.0;
    
    // 按中心点分左右
    NSMutableArray *leftViews  = [NSMutableArray array];
    NSMutableArray *rightViews = [NSMutableArray array];
    
    for (UIView *v in platterViews) {
        CGFloat centerX = v.frame.origin.x + v.frame.size.width / 2.0;
        if (centerX < midX) {
            [leftViews addObject:v];
        } else {
            [rightViews addObject:v];
        }
    }
    
    // 左侧:按 x 升序,从 0 开始依次排列
    [leftViews sortUsingComparator:^NSComparisonResult(UIView *a, UIView *b) {
        return a.frame.origin.x > b.frame.origin.x
            ? NSOrderedDescending : NSOrderedAscending;
    }];
    CGFloat leftX = 0;
    for (UIView *v in leftViews) {
        [self fixPlatterView:v toX:leftX];
        leftX += v.frame.size.width;
    }
    
    // 右侧:按 x 降序,从右边缘 -5 开始向左排列
    [rightViews sortUsingComparator:^NSComparisonResult(UIView *a, UIView *b) {
        return a.frame.origin.x < b.frame.origin.x
            ? NSOrderedDescending : NSOrderedAscending;
    }];
    CGFloat rightX = navBarWidth - 5;
    for (UIView *v in rightViews) {
        rightX -= v.frame.size.width;
        [self fixPlatterView:v toX:rightX];
    }
}

- (void)collectPlatterViews:(UIView *)view result:(NSMutableArray *)result {
    for (UIView *subview in view.subviews) {
        if ([NSStringFromClass(subview.class) containsString:@"PlatterView"]) {
            [result addObject:subview];
        } else {
            [self collectPlatterViews:subview result:result];
        }
    }
}

- (void)fixPlatterView:(UIView *)platterView toX:(CGFloat)x {
    // 优先修改约束
    for (NSLayoutConstraint *constraint in platterView.superview.constraints) {
        if (constraint.firstItem == platterView &&
            constraint.firstAttribute == NSLayoutAttributeLeading) {
            constraint.constant = x;
        }
    }
    // frame 兜底
    CGRect frame = platterView.frame;
    frame.origin.x = x;
    platterView.frame = frame;
}

调用时机

该方法需要在UINavigationBar布局完成后调用,推荐在 layoutSubviews 末尾触发:

- (void)layoutSubviews {
    [super layoutSubviews];
    
    if (@available(iOS 26.0, *)) {
        [self fixPlatterViewSpace];
    }
}

逻辑拆解

1. 递归收集 PlatterView

[self collectPlatterViews:self result:platterViews];

使用类名字符串匹配 PlatterView,而非直接引用私有类,规避了编译报错。找到 PlatterView 后立即收集,不再递归其子视图,防止嵌套层级的重复收集。

2. 以中线划分左右语义区

CGFloat midX = navBarWidth / 2.0;

导航栏天然地以中线分隔 leftBarButtonItemsrightBarButtonItems 的语义区域,以此作为分组依据,保证左右按钮的 PlatterView 不会被错误归类。

3. 左侧从 x=0 紧密排列

leftX = 0
[BackButton]x = 0
[OtherButton]x = BackButton.width

从导航栏左侧起点开始,将各 PlatterView 依次紧贴排列,彻底消除相邻玻璃背景块之间的系统默认间距,还原旧版左侧按钮的紧凑布局。

4. 右侧从右边缘留 5pt 向左排列

rightX = navBarWidth - 5
[Button2] → rightX -= Button2.width
[Button1] → rightX -= Button1.width

保留 5pt 右侧安全边距,确保最右侧玻璃背景块不会贴边,同时各 PlatterView 之间零间距紧密排布,与旧版右侧按钮排列保持一致。

5. 约束修改 + frame 双保险

// 先改约束(正确路径)
constraint.constant = x;
// 再改 frame(兜底)
platterView.frame = frame;

优先走 Auto Layout 路径修改 Leading 约束保证一致性,frame 赋值作为兜底,确保在纯 frame 布局场景下同样生效。


注意事项

事项 说明
仅限 iOS 26+ @available(iOS 26.0, *) 包裹调用,避免影响低版本行为
调用时机 必须在 layoutSubviews 之后,frame 确定后才能正确分组
Safe Area 左侧从 x=0 起排,刘海屏 / Dynamic Island 下需结合 safeAreaInsets.left 调整起始偏移
私有类名风险 依赖类名包含 PlatterView 的字符串匹配,若苹果后续改名则需同步更新
约束冲突 当前仅修改 Leading 约束;若 PlatterView 同时存在 Trailing / Center 约束,可能引发冲突,需一并处理

小结

iOS 26 的 Liquid Glass 设计语言改变了导航栏按钮的默认视觉风格。对于需要在 iOS 26 下维持旧版导航栏样式的项目,完整的适配路径分为两步:第一步通过 hidesSharedBackground = YES 拆分共享玻璃背景,让每个 item 独立渲染;第二步通过运行时遍历 PlatterView 并强制重置间距,将按钮排列收紧为旧版的紧凑样式。两步缺一不可。

我做了一个鼾声记录App,聊聊背后的功能设计

最近做了一款叫「睡眠声音日记」的App,主要用来记录睡眠时的鼾声和梦话。

今天主要聊聊这个App的功能设计思路。

为什么做这个App?

起因很简单:我自己打鼾,但完全不知道每晚打多少、什么时候最严重。 市面上的睡眠App大多侧重睡眠阶段分析,对鼾声的处理比较粗糙。我想做一个真正能听清楚每一段鼾声的工具。

核心:ML鼾声识别

App用CoreML跑了一个本地训练的声音分类模型,实时区分鼾声和人声(梦话)。每检测到一段就自动裁剪保存音频片段,第二天可以逐段回听。

不依赖网络,所有识别都在本地完成,隐私上比较放心。

灵敏度做了三档可调,适配不同噪音环境。

睡眠评分:5个维度

单纯告诉用户"你昨晚打了12次鼾"其实没什么指导意义,所以我做了一套100分制的评分系统,拆成5个维度:睡眠时长、鼾声/呼吸、深睡质量、睡眠连续性、身体恢复。每个维度单独打分,用户一眼就能看出问题出在哪。

AI个性化分析

接入了大模型做每日分析。不是泛泛的建议,而是把用户昨晚的实际数据(鼾声次数、时段分布、评分、HealthKit数据)传进去,生成针对性的建议。

历史页面还有基于多晚数据的趋势分析,能发现长期规律。如果鼾声连续多晚偏重,会主动建议用户去做专业评估。

趋势可视化

做了7晚和30晚两个维度的趋势图表:鼾声趋势、评分趋势、心率趋势、血氧趋势、睡眠时长柱状图。

还有一个昼夜节律分析,记录满5晚后自动解锁,分析用户的时型(早起型/夜猫子)。

这些图表对于观察干预效果很有用——比如换了枕头之后鼾声是不是真的少了。

Apple Watch用户体验拉满

如果你有Apple Watch,体验会更完整:

  • 手表上能看录音状态,直接停止记录
  • 昨晚的评分、鼾声、时长一目了然
  • 详情页有睡眠阶段时间线(核心/深睡/REM),鼾声事件直接叠在上面,一眼看出"你在深睡的时候鼾声最重"
  • 心率、血氧趋势图也有

小组件 + 灵动岛

桌面小组件做了3个尺寸,核心交互是一键开始/停止记录。大号组件额外展示鼾声时间分布图。录音期间支持灵动岛实时活动,锁屏上也能看到计时和事件计数。

其他细节

  • iCloud多设备同步
  • 数据备份恢复,支持导出
  • 音频自动清理(3/7/14/30天),重要片段可钉住跳过清理
  • 睡眠目标 + 睡前提醒
  • 成就系统,增加使用粘性
  • 一键生成分享图片,方便发给医生或朋友
  • iPad侧边栏适配

订阅模式

月订阅6元,年订阅38元,终身买断68元。

欢迎试用,有反馈随时评论区交流~

使用Wireshark进行TCP数据包抓包分析:三次握手与四次挥手详解

wireshark抓包分析TCP数据包

除了Wireshark,Sniffmaster作为一款全平台抓包工具,支持HTTPS、TCP和UDP协议,可在iOS、Android、Mac、Windows设备上实现无需代理、越狱或root的抓包操作,特别适合移动端和跨平台网络分析。

1、直接从TCP的 三次握手 开始说起

三次握手就是客户与服务器建立连接的过程

  • 客户向服务器发送SYN(SEQ=x)报文,然后就会进入SYN_SEND状态
  • 服务器收到SYN报文之后,回应一个SYN(SEQ=y)ACK(ACK=x+1)报文,然后就会进入SYN_RECV状态
  • 客户收到服务器的SYN报文,回应一个ACK(ACK=y+1)报文,然后就会进入Established状态

举例时间到!我们把客户端比作男生,服务器比作女生

第一次握手就像是男生对女生的告白:我喜欢你我们在一起吧。(之后,男孩就要等待女孩的回复,因为要确定女孩听到他说的话)

第二次握手则是女生的回应:好呀好呀。(之后,女孩也要等待,因为要确定男孩听到她的答复)

第三次握手就是男生的回应:真好,我们去吃火锅吧~。(此时,两人都确定对方收到了消息,关系成功建立)

也就是客户端和服务器数据的传输

接下来,我们抓包分析一下三次握手建立的过程

第一次握手:我向服务器发送了SYN,并设置Seq=0(x),请求与服务器建立连接

第二次握手:服务器向我回应了SYN,并设置Seq=0(y),ACK=1(x+1)

第三次握手:我收到服务器的SYN报文,回应一个ACK=1(y+1)

2、再接着说说四次挥手

四次挥手就是客户与服务器断开连接的过程

  • 客户发送一个FIN,断开与服务器的连接
  • 服务器收到FIN,回应一个ACK,确认序号为收到的序号加1
  • 服务器关闭客户端的连接,并发送一个FIN
  • 客户回发ACK确认,并将确认序号设置为收到序号加1

又到了举例时间!我们同样把客户端比作男生,服务器比作女生

第一次挥手:随着时间的流逝,女生变了,于是男生给女生发了分手短信,然后等待女生的回复

第二次挥手:女生听到后,伤心欲绝,就告诉男生:分手就分手,我把你的东西收拾收拾都还给你。男生就知道了女生同意了分手,于是等待女生把东西收拾好交还给他

第三次挥手:女生把男生的东西都收拾好,给男生发了第二条短信让他来取

第四次挥手:男生收到后,在回复最后一条短信,我知道了,我现在去取。于是关系断了

也就是客户端和服务器的连接中断

接下来,抓包看一下四次挥手的过程

第一次挥手:我向服务器发送FIN,Seq=3092,Ack=183

第二次挥手:服务器回发了ACK,Seq=183,Ack=3093

第三次挥手:服务器发送FIN,Seq=183,Ack=3093

第四次挥手:我向服务器回复了ACK,Seq=3093,Ack=184

3、TCP报文段格式分析

源端口和目的端口: 各占16位,这两个字段分别填入发送该报文段应用程序的源端口号和接收该报文段的应用程序的目的端口号

序列号: 占32位,TCP连接中传送的数据流中的每一个字节都编上一个序号,序号字段的值则指的是本报文段所发送的数据的第一个字节的序号

确认号: 占32位,表示期望收到对方下一个报文段的第一数据字节的序号。

数据偏移: 占4位,又称首部长度。指出首部的长度,即数据离开报文段开始的偏移量。

保留: 占6位,留待后用,目前置为0

标志: 占6位,又称控制字段,各位都有特定意义

  • 紧急URG,表示本报文数据的紧急程度,URG=1表示本报文具有高优先级
  • 确认ACK,ACK=1时,确认号字段才有意义
  • 推送PSH,PSH=1时,表示请求接收端TCP将本报文段立即送往其应用层
  • 复位RST,RST=1时,表示TCP连接中出现了严重错误,必须释放传输连接,而后在重建
  • 同步SYN,该位在连接建立时使用,起着序号同步的作用
  • 终止FIN,用来释放一个链接

窗口: 占16位,该字段用于流控制

校验和: 占16位,该字段的校验范围是整个报文段(包括首部和数据)

紧急指针: 占16位,当URG=1时有意义,指出紧急数据的末尾在报文段中的位置,使得接收端能知道紧急数据的字节数

选项与填充: 最长可达40B

Flutter iOS 包破解风险处理 可读信息抹除

Flutter 项目上线 iOS 后,如果有人拿到 IPA,第一步有可能不是反编译,而是直接解包。解压之后,目录结构非常清晰:Dart 代码、资源文件、插件模块都在不同位置。只要把这些信息拼起来,就能还原出应用的大致逻辑。

在一个包含会员系统和动态配置的 Flutter 项目中,我们专门做过一次抗破解处理。


先把 Flutter IPA 拆开看

构建完成 IPA 后,直接解压:

unzip Runner.ipa

进入目录:

Payload/Runner.app

可以看到几个关键内容:

App.framework
flutter_assets/
Frameworks/

进入 flutter_assets

assets/
isolate_snapshot_data
kernel_blob.bin

其中:

  • kernel_blob.bin:Dart 编译产物
  • assets/:资源文件
  • App.framework:部分逻辑代码

先处理 Dart 层(但不要停在这里)

Flutter 提供了混淆选项:

flutter build ios --obfuscate --split-debug-info=./symbols

执行后:

  • Dart 符号被替换
  • 生成符号映射文件

但这一步完成后,如果你再解包 IPA,会发现:

  • 资源名称仍然清晰
  • JS / JSON 可读
  • iOS 原生符号仍然存在

也就是说,这一步只是处理了 Dart 层。


处理 Flutter 资源目录(重点)

进入 flutter_assets/assets,如果看到类似:

images/vip_banner.png
config/payment.json
html/activity.html

这些名称已经足够说明业务结构。

我们做的处理是:不改 Flutter 工程,而是在 IPA 层统一修改

使用 Ipa Guard:

  • 导入 IPA
  • 切换到资源模块
  • 勾选图片、JSON、HTML、JS

资源混淆

执行后:

vip_banner.png → a8d3k.png
payment.json → x92ks.json

把 JS / HTML 再压一遍

如果 Flutter 中嵌入了 H5 页面(WebView),这些文件仍然是可读的。

在构建阶段或解包后处理:

terser main.js -o main.min.js

或者:

uglifyjs page.js -o page.min.js

处理后再放回 IPA,再用 Ipa Guard 改名。

这样做的结果是:

  • 内容压缩
  • 文件名无意义
  • 路径不可读

处理 iOS 原生层(很多人忽略)

Flutter 并不完全是 Dart,还包含:

  • 插件代码(Swift / OC)
  • 原生桥接层
  • SDK 逻辑

这些内容在 IPA 中属于 Mach-O 二进制。

检查一下:

strings AppBinary | grep Manager

如果看到:

FlutterPaymentManager
UserAuthHandler

说明原生层完全可读。


用 Ipa Guard 做二进制混淆

在代码模块中:

  • 选择 Swift 类
  • 选择 OC 方法
  • 勾选关键符号

代码混淆

执行后:

FlutterPaymentManager → k39sd2

再次查看:

strings AppBinary | grep Payment

已经找不到原始名称。


修改资源 MD5(解决“复用识别”问题)

如果多个应用使用同一套 UI 资源,即使改名也可能被识别。

Ipa Guard 提供 MD5 修改功能:

  • 图片内容不变
  • 文件指纹改变

md5修改

验证:

md5 vip_banner.png

处理前后不同。

这一步更多是避免资源被简单比对。


删掉那些“多余信息”

Flutter 构建过程中,有时会带入调试信息。

可以检查:

strings AppBinary | grep Flutter

如果输出包含日志或调试字段,可以在 IPA 处理阶段清理。

Ipa Guard 支持删除部分调试信息。


签名并直接安装测试

所有修改完成后,必须重新签名。

可以使用:

kxsign sign app.ipa \
-c cert.p12 \
-p password \
-m dev.mobileprovision \
-z test.ipa \
-i

或者直接在 Ipa Guard 中配置证书。

设备连接后可以直接安装。 重签名


测试关注点(Flutter 特有)

Flutter 项目测试时,需要特别看:

  • 页面渲染是否正常
  • Dart 调用是否异常
  • 插件是否还能调用
  • WebView 是否加载成功

如果某些页面加载失败,基本可以定位到资源路径被误处理。


Flutter iOS 包的破解入口并不只有 Dart 代码。资源目录、JS 文件、原生模块符号,这些地方同样可以被利用。单一手段很难覆盖所有暴露点。

在实际项目中,通过 Flutter 构建参数处理 Dart 层,再结合 Ipa Guard 对 IPA 进行资源混淆、二进制符号处理和 MD5 修改,可以在不侵入项目结构的情况下完成一轮补强。

参考链接:ipaguard.com/blog/159

ObservableObject @Published @ObservedObject那些事

先理解这三个为什么要一起讲

它们是一套组合拳,缺一不可:

角色 是什么
ObservableObject 一个协议,贴在 class 上,宣告"我是可被观察的数据源"
@Published 一个 Property Wrapper,贴在属性上,宣告"这个属性变化时要通知订阅者"
@ObservedObject 一个 Property Wrapper,贴在 View 的属性上,宣告"我订阅这个数据源,它变化我就刷新"

为什么需要这套东西?@State 不够用吗?

@State 适合简单的值类型,但现实中你的数据模型往往是一个 class,有很多属性和方法,且需要被多个平级 View 共享

// 一个用户信息模型,多个页面都要用
class UserModel {
   var name: String = "Tom"
   var age: Int = 18
   var score: Int = 0
   // ... 还有很多方法
}

把这个 class 塞进 @State 是行不通的——@State 是为值类型设计的,对 class 的引用地址变化不敏感,属性改了 UI 也不会刷新。


三件套的用法

// 第一步:让你的 class 遵守 ObservableObject 协议
class UserModel: ObservableObject {
   // 第二步:在需要触发 UI 刷新的属性上加 @Published
   @Published var name: String = "Tom"
   @Published var score: Int = 0
   var internalCache: String = ""  // 不加 @Published,改它不会刷新 UI
}

// 第三步:在 View 里用 @ObservedObject 订阅这个模型
struct ProfileView: View {
   @ObservedObject var user: UserModel

   var body: some View {
       VStack {
           Text(user.name)
           Text("\(user.score)")
           Button("加分") {
               user.score += 1   // 改 @Published 属性 → 触发 UI 刷新
           }
       }
   }
}

// 使用:顶层 View 用 @StateObject 持有并创建模型
struct ContentView: View {
   @StateObject var user = UserModel()

   var body: some View {
       ProfileView(user: user)
   }
}

三件套的本质

@Published 本质上是:

@propertyWrapper
public struct Published<Value> {
   // 每次 wrappedValue 被 set,就通过 objectWillChange 发出通知
   public var wrappedValue: Value
   // $score 拿到的是一个 Combine Publisher,可以接链式操作
   public var projectedValue: Publisher
}

ObservableObject 协议本质上是:

public protocol ObservableObject: AnyObject {
   // 编译器会自动合成这个,你的 @Published 属性改变时,它会发出信号
   var objectWillChange: ObservableObjectPublisher { get }
}

@ObservedObject 本质上是:View 订阅了 user.objectWillChange,只要它 emit,SwiftUI 就重新计算这个 View 的 body。

整个流程: user.score += 1@Published 的 setter 触发 → user.objectWillChange.send() → 订阅了它的 @ObservedObject 感知到 → SwiftUI 重新渲染对应的 View


@ObservedObject vs @StateObject

这是一个非常容易踩的坑:

@ObservedObject @StateObject
数据归属 不拥有,由外部传入 拥有,由这个 View 创建和持有
生命周期 跟随外部,不负责销毁 跟随 View,View 消失时销毁
典型场景 子 View 接收父 View 传来的模型 根 View 或顶层 View 创建模型

经验法则:谁创建,谁用 @StateObject;谁接收,谁用 @ObservedObject


使用时需要关心的问题

  1. 只有 class 能用ObservableObjectAnyObject 的子协议,struct 和 enum 无法遵守,这套机制天生是为引用类型设计的。

  2. @Published 要精准:不是所有属性都需要 @Published,只给真正需要驱动 UI 的属性加,滥加会导致不必要的 View 重渲染,影响性能。

  3. objectWillChange 是"将要改变":SwiftUI 在属性改变之前就会收到通知,你通常不需要手动调用它,但在某些手动控制的场景可以用 objectWillChange.send() 主动触发刷新。

@Binding 的那些事

先理解 @Binding 解决什么问题

@State 的时候,状态归属于某一个 View。但子 View 怎么修改父 View 的状态?

struct ParentView: View {
    @State var isOn: Bool = false

    var body: some View {
        ToggleView(isOn: isOn) // ❌ 子 View 拿到的只是一个值的拷贝
    }
}

你把 isOn 传给子 View,子 View 改了它自己的拷贝,父 View 毫不知情,UI 也不会更新。


@Binding 就是用来解决这个问题的

@Binding 不是一份数据的拷贝,而是一条双向通道,指向原始数据的存储位置。 读它,读的是原始值;写它,写的是原始存储,父 View 会同步感知并刷新。

// 父 View:状态归属于这里
struct ParentView: View {
    @State var isOn: Bool = false

    var body: some View {
        // 用 $ 前缀把 @State 转成 Binding 传下去
        ToggleView(isOn: $isOn)
    }
}

// 子 View:不拥有状态,只拿到一条"通道"
struct ToggleView: View {
    @Binding var isOn: Bool  // 声明为 Binding,表示"我不拥有这个数据"

    var body: some View {
        Button("切换") {
            isOn.toggle()   // 写的是父 View 里的原始 @State,触发父 View 刷新
        }
    }
}

@Binding 的本质

@propertyWrapper
public struct Binding<Value> {
    // 你平时用 isOn 读写的就是这个
    public var wrappedValue: Value { get nonmutating set }

    // 你用 $isOn 拿到的还是 Binding 自身,可以继续往下传
    public var projectedValue: Binding<Value> { get }
}

@Binding 内部存的不是值本身,而是一对 getter + setter 闭包,分别指向上层 @State(或其他数据源)的读写操作。所以写 isOn = true 时,实际上是调用了那个 setter 闭包,最终改变的是父 View 的 @State


使用 @Binding 时需要关心的问题

  1. 数据归属权问题@Binding 的原则是"我不拥有数据,我只是一个读写通道"。如果一个 View 需要拥有状态,用 @State;如果只是借用和修改上层的状态,用 @Binding

  2. 单向来源原则(Single Source of Truth):一条 @Binding 链条最终必须溯源到某个真实的数据存储(比如 @State@StateObject 中的属性),不要出现 Binding 套 Binding 套 Binding 的迷宫,链条越短越清晰。

  3. $ 符号的含义$isOn 拿到的是 projectedValue,对 @State 来说它是一个 Binding<Bool>,这就是为什么父 View 传 $isOn,而子 View 声明 @Binding var isOn,类型是对得上的。

  4. 不要在 body 外部调用:和 @State 一样,对 @Binding 属性的读写应发生在 bodybody 调用的方法中,以确保 SwiftUI 能正确追踪依赖。

Xcode 26.4 AFNetworking 私有头文件报错处理记录

问题现象

在当前工程执行 Pods 编译时,AFNetworking 4.0.1 出现以下报错:

/Pods/AFNetworking/AFNetworking/AFNetworkReachabilityManager.m:26:9 Use of private header from outside its module: 'netinet6/in6.h'
/Pods/AFNetworking/AFNetworking/AFHTTPSessionManager.m:32:9 Use of private header from outside its module: 'netinet6/in6.h'

根因分析

AFNetworking 4.0.1 在以下源码中直接引用了私有头文件 #import <netinet6/in6.h>

  • Pods/AFNetworking/AFNetworking/AFNetworkReachabilityManager.m
  • Pods/AFNetworking/AFNetworking/AFHTTPSessionManager.m

新版 Xcode / Clang 会将该引用识别为“模块外私有头文件访问”,从而在编译阶段直接报错。

这里并不需要通过放宽全局编译限制来绕过问题。对应文件实际依赖的 IPv6 结构体和常量可由公开头 #import <netinet/in.h> 提供,因此直接移除 #import <netinet6/in6.h> 即可。

处理方案

为了避免每次 pod install 后手动修改 Pods 目录,本次将修复逻辑固化到 Podfilepost_install 阶段。

处理原则如下:

  1. 扫描 AFNetworking 目录下所有 .h/.m 文件。
  2. 查找 #import <netinet6/in6.h>
  3. 如果存在,则自动删除该导入。
  4. 由于 Pods 内目标文件可能是只读权限,写入前临时补充写权限,写入后恢复原权限。

最终 Podfile 补丁

本次在 Podfile 中新增以下逻辑:

def patch_afnetworking_private_header(installer)
  # 扫描并移除 AFNetworking 对私有 IPv6 头文件的直接引用,兼容新版 Xcode 的模块校验。
  afnetworking_dir = File.join(installer.sandbox.pod_dir('AFNetworking'), 'AFNetworking')
  return unless Dir.exist?(afnetworking_dir)

  private_header_import = '#import <netinet6/in6.h>'
  Dir.glob(File.join(afnetworking_dir, '**', '*.{h,m}')).each do |file_path|
    next unless mcs_file_exists(file_path)

    file_content = File.read(file_path)
    next unless file_content.include?(private_header_import)

    original_mode = File.stat(file_path).mode
    File.chmod(original_mode | 0o200, file_path)
    File.write(file_path, file_content.gsub(private_header_import, ''))
    File.chmod(original_mode, file_path)
    puts "patched AFNetworking private header import: #{File.basename(file_path)}"
  end
end

post_install do |installer|
  patch_afnetworking_private_header(installer)

  installer.pods_project.targets.each do |target|
    target.build_configurations.each do |config|
      config.build_settings['IPHONEOS_DEPLOYMENT_TARGET'] = '9.0'
      config.build_settings["EXCLUDED_ARCHS[sdk=iphonesimulator*]"] = "arm64"
      config.build_settings['CLANG_ENABLE_OBJC_WEAK'] = 'YES'
      config.build_settings['SWIFT_VERSION'] = '5.0'
    end
  end
end

实际修复结果

执行 pod install 后,以下两个文件中的私有头导入已被移除:

  • Pods/AFNetworking/AFNetworking/AFNetworkReachabilityManager.m
  • Pods/AFNetworking/AFNetworking/AFHTTPSessionManager.m

修复后的关键导入区如下:

#import <netinet/in.h>

#import <arpa/inet.h>
#import <ifaddrs.h>
#import <netdb.h>

验证过程

1. 安装 Pods

执行:

pod install

结果:

  • Pod installation complete!
  • post_install 补丁正常执行。

2. 检查私有头是否已全部移除

执行:

rg -n "netinet6/in6.h" Pods/AFNetworking/AFNetworking

结果:

  • 无输出。
  • 说明 AFNetworking 目录下已不存在该私有头引用。

3. 单独编译 AFNetworking Target

执行:

xcodebuild -project Pods/Pods.xcodeproj \
  -scheme AFNetworking \
  -configuration Debug \
  -sdk iphonesimulator \
  -derivedDataPath /tmp/TXLAPP_IOS_Pods_DerivedData \
  CODE_SIGNING_ALLOWED=NO build

结果:

** BUILD SUCCEEDED **

说明:

  • AFNetworking 已成功完成真实编译。
  • AFHTTPSessionManager.mAFNetworkReachabilityManager.m 均已通过编译。
  • 本次处理的私有头报错已被清除。

影响文件

  • Podfile
  • Podfile.lock
  • Pods/Manifest.lock
  • Pods/AFNetworking/AFNetworking/AFNetworkReachabilityManager.m
  • Pods/AFNetworking/AFNetworking/AFHTTPSessionManager.m

注意事项

  1. 这次修复是对第三方库源码的安装阶段补丁,不建议直接长期手改 Pods 文件后不保留 Podfile 逻辑。
  2. 后续只要重新执行 pod install,该补丁就会再次自动生效。
  3. 当前 Podfile 里仍将 Pods 的 IPHONEOS_DEPLOYMENT_TARGET 设置为 9.0,在新 Xcode 下会出现最低部署版本警告;这不是本次私有头报错的根因,但后续可以单独再整理处理。

结论

本次问题的本质是旧版 AFNetworking 源码中引用了私有系统头文件,导致新编译环境不再允许通过。采用 Podfile post_install 自动移除 netinet6/in6.h 引用的方式,可以稳定修复该问题,并保证后续重新安装 Pods 时无需重复人工处理。

@state的一些琐事

先理解 Property Wrapper 是什么

@propertyWrapper 就是让你可以自定义 @ 修饰符的机制。 @State@Binding 这些不是Swift内置的魔法,它们本质上就是普通的 struct,只不过被 @propertyWrapper 修饰了,所以才能用 @ 语法来用。 能理解吗?是不是还是很难理解,没事我写一个例子你就能理解了

假设你有一个属性,每次读取它都想打印一条日志:
var age: Int = 18
var age: Int = 18 { 
    didSet { print("age 变了,新值是 \(age)") }
}
但如果你有 100 个属性都需要这个功能呢?你要写 100 次 `didSet`?

Property Wrapper 就是用来解决这个问题的

你可以把"通用的包装逻辑"封装起来,然后像帖标签一样贴到任何属性上。

// 第一步:定义一个 Property Wrapper
@propertyWrapper
struct Logged {
    private var value: Int
    // initialValue 参数后面可以跟很多参数,自定义
    init(initialValue: Int) {
        self.value = initialValue
    }
    
    var wrappedValue: Int {
    //这里的get 和set 我们可以自定义任何我们想要的操作,比如有多个参数我们可以把这些参数拼接起来返回等等
        get { value }
        set {
            print("值变了,新值是 \(newValue)")  // 通用逻辑写在这里
            value = newValue
        }
    }
}

// 第二步:像贴标签一样使用它
@Logged var age = 18
@Logged var score = 100

// 现在 age 和 score 改变时,都会自动打印日志
age = 20   // 打印:值变了,新值是 20
score = 99 // 打印:值变了,新值是 99

所以 @propertyWrapper 本质上就是

把"对属性的操作逻辑"打包成一个 struct,然后用 @ 语法贴到属性上,让这个属性自动拥有那些逻辑。

回到 @State

@State 干的事情无非就是:

@propertyWrapper
public struct State<Value> {  
    // 1. 让你能直接赋初始值
    public init(initialValue value: Value)   
    // 2. 你平时用 brain 读写的就是这个 (这里set 之后苹果偷偷的去给你刷新了UI)
    public var wrappedValue: Value { get nonmutating set }    
    // 3. 你用 $brain 拿到的就是这个(一个 Binding)
    public var projectedValue: Binding<Value> { get }
}

@State 非常适合 struct 或者 enum 这样的值类型,它可以自动为我们完成从状态 到 UI 更新等一系列操作。但是它本身也有一些限制,我们在使用 @State 之前,对 于需要传递的状态,最好关心和审视下面这两个问题:

  1. 这个状态是属于单个 View 及其子层级,还是需要在平行的部件之间传递和使 用?@State 可以依靠 SwiftUI 框架完成 View 的自动订阅和刷新,但这是有 条件的:对于 @State 修饰的属性的访问,只能发生在 body 或者 body 所调 用的方法中。你不能在外部改变 @State 的值,它的所有相关操作和状态改变 都应该是和当前 View 挂钩的。如果你需要在多个 View 中共享数据,@State 可能不是很好的选择;如果还需要在 View 外部操作数据,那么 @State 甚至 就不是可选项了。
  2. 状态对应的数据结构是否足够简单?对于像是单个的 Bool 或者 String, @State 可以迅速对应。含有少数几个成员变量的值类型,也许使用 @State 也还不错。但是对于更复杂的情况,例如含有很多属性和方法的类型,可能其 中只有很少几个属性需要触发 UI 更新,也可能各个属性之间彼此有关联,那 么我们应该选择引用类型和更灵活的可自定义方式。

Xcode MCP Server 完全指南:从智能配置到编程控制

目录概要


为什么需要 MCP?

如果你用过 Cursor 或 Claude 写代码,一定有过这样的体验:AI 能侃侃而谈生成代码,但真要它帮你跑个测试、修个编译错误,它就傻眼了——因为它"看不见"你的 Xcode 工程,也"摸不着"编译器。

Model Context Protocol (MCP) 就是来解决这个问题的。它像一根 USB 线,把 AI 助手和 Xcode 连接起来,让 AI 可以直接读取文件、运行构建、执行测试,甚至渲染 SwiftUI 预览。换句话说,MCP 让 Xcode 变成了一个可编程的"智能引擎"。

本文将从系统配置工具实战,带你完整掌握 Xcode MCP Server 的使用。文章后半部分,我还会穿插一些编译器演进的历史故事——毕竟,理解了"从哪里来",才更能明白"往哪里去"。

graph LR
    subgraph AI["🤖 AI 助手"]
        A1[Cursor]
        A2[Claude CLI]
        A3[Codex]
    end

    subgraph MCP["🔌 MCP 桥梁"]
        B[mcpbridge]
    end

    subgraph Xcode["🛠️ Xcode 引擎"]
        C1[📄 文件读写]
        C2[🔨 编译构建]
        C3[✅ 测试运行]
        C4[👁️ UI 预览]
        C5[🔍 代码搜索]
    end

    A1 --> B
    A2 --> B
    A3 --> B
    B --> C1
    B --> C2
    B --> C3
    B --> C4
    B --> C5

    style AI fill:#e1f5fe,stroke:#01579b,stroke-width:2px
    style MCP fill:#fff9c4,stroke:#fbc02d,stroke-width:2px
    style Xcode fill:#e8f5e9,stroke:#2e7d32,stroke-width:2px

一、系统与环境前置要求

要玩转 Xcode Intelligence 和 MCP,硬件和系统是第一道门槛。Apple 这次把门槛卡得很死:

  • macOS:必须运行 macOS Sequoia 15.2 或更高版本。别问为什么,问就是 Apple Intelligence 强依赖于端侧 NPU 算力。
  • 硬件:必须使用 Apple Silicon (M1 及后续芯片) 的 Mac。Intel 用户暂时只能眼馋。
  • XcodeXcode 26.3 或更高版本。这个版本内置了 mcpbridge 工具,也就是 MCP 的服务端。

另外,需要在 系统设置 > Apple Intelligence & Siri 中确保开关已开启。Xcode 的智能功能是 Apple Intelligence 的一部分,系统层面不开,Xcode 里也开不了。

graph TD
    subgraph 前置检查清单
        direction TB
        H["💻 硬件检查"] --> H1{"Apple Silicon?<br/>M1 / M2 / M3 / M4"}
        H1 -->|✅ 是| S["🖥️ 系统检查"]
        H1 -->|❌ Intel| FAIL["⛔ 不支持"]
        S --> S1{"macOS Sequoia 15.2+?"}
        S1 -->|✅ 是| X["📱 Xcode 检查"]
        S1 -->|❌ 版本过低| UPDATE1["⬆️ 升级 macOS"]
        X --> X1{"Xcode 26.3+?"}
        X1 -->|✅ 是| AI["🧠 Apple Intelligence"]
        X1 -->|❌ 版本过低| UPDATE2["⬆️ 升级 Xcode"]
        AI --> AI1{"系统设置中<br/>Apple Intelligence 已开启?"}
        AI1 -->|✅ 是| OK["🎉 环境就绪!"]
        AI1 -->|❌ 未开启| ENABLE["⚙️ 前往设置开启"]
    end

    style FAIL fill:#ffcdd2,stroke:#c62828
    style OK fill:#c8e6c9,stroke:#2e7d32
    style UPDATE1 fill:#fff9c4,stroke:#f57f17
    style UPDATE2 fill:#fff9c4,stroke:#f57f17
    style ENABLE fill:#fff9c4,stroke:#f57f17

二、开启 Xcode Intelligence:模型提供商配置

Xcode 26.3 的智能功能采用了插件化的模型提供商架构(Provider Architecture),你可以同时接入多个模型源,根据任务需求灵活切换。

在 Xcode 中打开 Settings (⌘,) > Intelligence,你会看到三个主要提供商:

A. Apple (本地/云端混合)

  • 默认集成,无需额外配置。
  • 提供基础的代码补全(Predictive Code Completion)和轻量级重构建议,针对 Swift 和 Apple SDK 有优化。

B. ChatGPT (OpenAI)

  • 点击 ChatGPT in Xcode 下的 Turn On。
  • 绑定 ChatGPT 账号(支持 Free 和 Plus)。
  • 在 Project Editor 中,可以为特定 Target 选择模型的 Reasoning Level(推理等级),控制生成代码的深度。

C. Claude (Anthropic)

  • 点击 Claude 下的 Sign In 授权。
  • 如果安装了 Claude Agent 组件,可以在此配置其构建和测试权限。

小贴士:Xcode 内置的 Agent 配置目录位于 ~/Library/Developer/Xcode/CodingAssistant/,与标准的 .codex.claude 配置独立,所以不会干扰你现有的命令行工具配置。

D. 关键一步:启用 Xcode Tools MCP Server

在 Intelligence 设置页面的最底部,找到 Model Context Protocol 区域,将 Xcode Tools 的开关拨至 ON

技术原理:开启后,Xcode 主进程会启动一个名为 mcpbridge 的 XPC 服务,监听来自外部工具的连接请求。当外部工具首次尝试连接时,Xcode 会弹出权限确认对话框——务必点击 Allow,否则一切免谈。

graph TB
    subgraph XcodeSettings["⚙️ Xcode Settings > Intelligence"]
        direction TB
        P1["🍎 Apple<br/>━━━━━━━━━━<br/>本地 + 云端混合<br/>代码补全 / 重构建议<br/>🟢 默认开启"]
        P2["🤖 ChatGPT (OpenAI)<br/>━━━━━━━━━━<br/>Turn On → 绑定账号<br/>支持 Free / Plus<br/>可调 Reasoning Level"]
        P3["🟣 Claude (Anthropic)<br/>━━━━━━━━━━<br/>Sign In → 授权<br/>Claude Agent 构建权限<br/>独立配置目录"]
        P4["🔌 MCP Server<br/>━━━━━━━━━━<br/>Xcode Tools → ON<br/>启动 mcpbridge XPC<br/>⚠️ 首次连接需 Allow"]
    end

    P1 --- P2
    P2 --- P3
    P3 --- P4

    P4 -->|开启后| XPC["mcpbridge<br/>XPC 服务启动"]
    XPC -->|外部工具连接| ALLOW{"权限弹窗<br/>Allow?"}
    ALLOW -->|✅ Allow| READY["🎉 MCP 就绪"]
    ALLOW -->|❌ Deny| BLOCKED["⛔ 连接被拒"]

    style P4 fill:#fff9c4,stroke:#fbc02d,stroke-width:2px
    style READY fill:#c8e6c9,stroke:#2e7d32
    style BLOCKED fill:#ffcdd2,stroke:#c62828

三、MCP 架构揭秘:桥梁是如何搭建的

MCP 是一个标准化的协议,旨在解决 AI 模型与本地开发环境"隔离"的问题。它的架构可以用以下流程图清晰地表示:

graph TD
    subgraph Client["🌐 MCP 客户端"]
        A1["Claude Code<br/>(CLI)"]
        A2["Cursor<br/>(IDE)"]
        A3["Codex<br/>(CLI)"]
    end

    subgraph Bridge["🌉 MCP Bridge 层"]
        B["xcrun mcpbridge<br/>━━━━━━━━━━━━━<br/>协议: stdio JSON-RPC<br/>角色: 翻译官"]
    end

    subgraph XcodeProcess["🏗️ Xcode 主进程"]
        C["Xcode App<br/>━━━━━━━━━━━━━<br/>通信: XPC"]
        C --> D["🔨 Build System<br/>增量编译 / 错误诊断"]
        C --> E["📝 Source Editor<br/>文件读写 / 代码分析"]
        C --> F["🧪 XCTest Runner<br/>测试运行 / 结果收集"]
        C --> G["👁️ Preview Engine<br/>SwiftUI 预览渲染"]
        C --> H["📚 Documentation<br/>Apple 文档搜索"]
    end

    A1 -->|"stdio<br/>JSON-RPC"| B
    A2 -->|"stdio<br/>JSON-RPC"| B
    A3 -->|"stdio<br/>JSON-RPC"| B
    B -->|"XPC<br/>进程间通信"| C

    style Client fill:#e1f5fe,stroke:#01579b,stroke-width:2px
    style Bridge fill:#fff9c4,stroke:#fbc02d,stroke-width:2px
    style XcodeProcess fill:#d1c4e9,stroke:#512da8,stroke-width:2px

关键路径解读

  • mcpbridge 是一个命令行工具,通过 xcrun mcpbridge 启动。
  • 它与 Xcode 主进程通过 XPC 通信,调用 Xcode 内部的构建、编辑、调试等 API。
  • 外部客户端(如 Cursor、Claude CLI)通过**标准输入输出(stdio)**与 mcpbridge 交互,协议基于 JSON-RPC
  • 简单来说,mcpbridge 就是那个"翻译官",把 AI 的意图翻译成 Xcode 能懂的操作。
sequenceDiagram
    participant AI as 🤖 AI 助手 (Cursor)
    participant MCP as 🌉 mcpbridge
    participant Xcode as 🛠️ Xcode

    AI->>MCP: JSON-RPC 请求<br/>"BuildProject"
    MCP->>Xcode: XPC 调用<br/>触发编译
    Xcode-->>Xcode: 执行增量构建...
    Xcode->>MCP: XPC 响应<br/>编译结果 + 错误列表
    MCP->>AI: JSON-RPC 响应<br/>结构化错误信息

    Note over AI: 解析错误,定位文件和行号

    AI->>MCP: JSON-RPC 请求<br/>"XcodeRead" 读取错误文件
    MCP->>Xcode: XPC 调用
    Xcode->>MCP: 文件内容
    MCP->>AI: 带行号的源码

    Note over AI: 分析问题,生成修复代码

    AI->>MCP: JSON-RPC 请求<br/>"XcodeUpdate" 修复代码
    MCP->>Xcode: XPC 调用
    Xcode->>MCP: 更新成功
    MCP->>AI: 确认响应

四、客户端接入:让 Cursor/Claude 操作 Xcode

场景 A:命令行工具 (Claude CLI / Codex)

Claude Code

claude mcp add --transport stdio xcode -- xcrun mcpbridge

Codex

codex mcp add xcode -- xcrun mcpbridge

场景 B:集成开发环境 (Cursor / Trae)

在编辑器的 MCP 配置文件中添加 Server 定义。

GUI 方式:进入 Settings > Features > MCP,点击 + Add New MCP Server。

Name: xcode
Transport: stdio
Command: xcrun mcpbridge

JSON 方式:修改配置文件 ~/.cursor/mcp.json~/.config/trae/mcp.json

{
  "mcpServers": {
    "xcode": {
      "command": "xcrun",
      "args": ["mcpbridge"]
    }
  }
}

注意mcpbridge 会自动检测当前运行的 Xcode 进程 ID(PID),一般无需手动指定环境变量。如果 Xcode 没打开,连接会失败——这是最常见的坑。

场景 C:项目级上下文提示

在项目根目录添加 AGENTS.mdCLAUDE.md 文件,里面可以写清楚:

  • 核心 Scheme 名称
  • 主要的 Test Plan
  • 特殊的构建脚本路径
  • 架构模式(MVVM/TCA 等)

MCP Client 会优先读取这些文件作为 System Prompt 的一部分,让 AI 更懂你的项目结构。

graph LR
    subgraph CLI["💻 命令行接入"]
        C1["claude mcp add<br/>--transport stdio<br/>xcode -- xcrun mcpbridge"]
        C2["codex mcp add<br/>xcode -- xcrun mcpbridge"]
    end

    subgraph IDE["🖥️ IDE 接入"]
        I1["Cursor<br/>~/.cursor/mcp.json"]
        I2["Trae<br/>~/.config/trae/mcp.json"]
    end

    subgraph Context["📋 项目上下文"]
        X1["AGENTS.md"]
        X2["CLAUDE.md"]
    end

    CLI --> MCP["🔌 mcpbridge"]
    IDE --> MCP
    Context -.->|"System Prompt"| MCP
    MCP --> Xcode["🛠️ Xcode"]

    style CLI fill:#e3f2fd,stroke:#1565c0
    style IDE fill:#f3e5f5,stroke:#7b1fa2
    style Context fill:#e8f5e9,stroke:#2e7d32
    style MCP fill:#fff9c4,stroke:#fbc02d,stroke-width:2px

五、MCP 工具集详解(20 个工具分类说明)

xcrun mcpbridge 暴露了 20 个核心工具,覆盖了从文件操作到测试运行的方方面面。下面按功能分类逐一说明。

graph TB
    subgraph Tools["🧰 MCP 工具集 - 20 个工具"]
        direction TB
        subgraph FileOps["📄 文件操作 (6)"]
            F1["XcodeRead"]
            F2["XcodeWrite"]
            F3["XcodeUpdate"]
            F4["XcodeMV"]
            F5["XcodeRM"]
            F6["XcodeMakeDir"]
        end
        subgraph SearchOps["🔍 代码搜索 (3)"]
            S1["XcodeGrep"]
            S2["XcodeGlob"]
            S3["XcodeLS"]
        end
        subgraph BuildOps["🔨 构建诊断 (4)"]
            B1["BuildProject"]
            B2["GetBuildLog"]
            B3["XcodeListNavigatorIssues"]
            B4["XcodeRefreshCodeIssuesInFile"]
        end
        subgraph TestOps["✅ 测试运行 (3)"]
            T1["GetTestList"]
            T2["RunSomeTests"]
            T3["RunAllTests"]
        end
        subgraph PreviewOps["👁️ 预览运行时 (2)"]
            P1["RenderPreview"]
            P2["ExecuteSnippet"]
        end
        subgraph DocOps["📚 文档窗口 (2)"]
            D1["DocumentationSearch"]
            D2["XcodeListWindows"]
        end
    end

    style FileOps fill:#e3f2fd,stroke:#1565c0
    style SearchOps fill:#fff3e0,stroke:#e65100
    style BuildOps fill:#fce4ec,stroke:#c62828
    style TestOps fill:#e8f5e9,stroke:#2e7d32
    style PreviewOps fill:#f3e5f5,stroke:#7b1fa2
    style DocOps fill:#efebe9,stroke:#4e342e

1. 文件操作类

工具 功能 关键参数
XcodeRead 读取文件内容,支持分页(limit/offset),最多 600 行 filePath, offset, limit
XcodeWrite 创建新文件或覆盖现有文件,自动添加到工程组 filePath, content
XcodeUpdate 增量编辑(基于字符串替换),比全量重写更省 token filePath, edits
XcodeMV 移动或重命名文件,保持工程结构一致性 sourcePath, destPath
XcodeRM 删除文件 filePath
XcodeMakeDir 创建目录 path

最佳实践:务必使用 XcodeRead 返回的行号作为参考,避免后续编辑时行号偏移。路径格式示例:MyProject/ViewControllers/MyViewController.swift

2. 代码搜索类

工具 功能 关键参数
XcodeGrep 在工程中搜索文本模式(支持正则) pattern, path, glob, type, outputMode
XcodeGlob 基于 glob 模式列出文件(如 **/*.swift pattern
XcodeLS 列出目录内容,类似 ls 命令 path

3. 构建与诊断类

工具 功能 关键返回值
BuildProject 触发当前 Scheme 的增量构建(阻塞调用) buildResult, errors[], elapsedTime
GetBuildLog 获取最近一次构建的详细日志 log
XcodeListNavigatorIssues 获取 Issue Navigator 中的实时问题(无需完整构建) issues 列表
XcodeRefreshCodeIssuesInFile 强制刷新并检索特定文件的编译器诊断 filePath 对应的诊断信息

4. 测试运行类

工具 功能 关键返回值
GetTestList 获取所有可用测试的层级结构(Test Plan → Class → Method) tests 层级列表
RunSomeTests 运行指定的测试用例 指定测试的结果
RunAllTests 运行当前 Scheme 中的所有测试 counts, results[](最多 100 条)

测试标识符示例MyProjectTests/UserProfileTests/testUserNameValidation

5. 预览与运行时

工具 功能 关键参数
RenderPreview 构建并渲染 SwiftUI 预览(#PreviewPreviewProvider),返回图片路径 sourceFilePath, timeout
ExecuteSnippet 在目标文件的上下文中动态执行 Swift 代码,类似 LLDB 的 expression 代码片段

6. 文档与窗口

工具 功能
DocumentationSearch 搜索 Xcode 内置的 Apple 开发文档和 WWDC 视频
XcodeListWindows 列出当前打开的 Xcode 窗口信息

注意:所有工具的参数中,tabIdentifier 通常可以省略,mcpbridge 会自动关联当前活跃的 Xcode 窗口。


六、实战场景与最佳实践

场景 1:修复编译错误

// 1. 构建并获取错误列表
BuildProject()

// 2. 读取有错误的文件
XcodeRead(filePath: "MyProject/ViewControllers/MyViewController.swift")

// 3. 修改文件
XcodeWrite(filePath: "MyProject/ViewControllers/MyViewController.swift", content: "修正后的代码...")

// 4. 再次构建验证
BuildProject()

场景 2:代码搜索与重构

// 搜索所有调用某个类的地方
XcodeGrep(pattern: "MyViewController", outputMode: "filesWithMatches")

// 逐个读取文件并修改
XcodeRead(filePath: "MyProject/ViewControllers/AnotherViewController.swift")
XcodeUpdate(...)

场景 3:测试驱动开发

// 获取测试列表
GetTestList()

// 运行指定测试
RunSomeTests(testIdentifiers: ["MyProjectTests/UserProfileTests/testUserInitialization"])

// 如果失败,修复后再运行

场景 4:UI 预览验证

// 修改 SwiftUI 代码后
XcodeWrite(filePath: "MyProject/Views/ProfileView.swift", content: "更新后的预览代码...")

// 渲染预览
RenderPreview(sourceFilePath: "MyProject/Views/ProfileView.swift")

最佳工作流建议

graph LR
    A["📖 读<br/>XcodeRead"] --> B["✏️ 写<br/>XcodeUpdate"]
    B --> C["🔨 验<br/>BuildProject"]
    C --> D{"编译通过?"}
    D -->|❌ 失败| A
    D -->|✅ 通过| E["🔍 搜<br/>XcodeGrep"]
    E --> F["🧪 测<br/>RunSomeTests"]
    F --> G{"测试通过?"}
    G -->|❌ 失败| A
    G -->|✅ 通过| H["👁️ 看<br/>RenderPreview"]
    H --> I["🎉 完成"]

    style A fill:#e3f2fd,stroke:#1565c0
    style B fill:#fff3e0,stroke:#e65100
    style C fill:#fce4ec,stroke:#c62828
    style D fill:#f5f5f5,stroke:#616161
    style E fill:#f3e5f5,stroke:#7b1fa2
    style F fill:#e8f5e9,stroke:#2e7d32
    style G fill:#f5f5f5,stroke:#616161
    style H fill:#ede7f6,stroke:#311b92
    style I fill:#c8e6c9,stroke:#2e7d32,stroke-width:3px

七、踩坑指南:解决 Cursor 兼容性问题

在 Xcode 26.3 (26.3 RC) 中,xcrun mcpbridge 返回的响应不完全符合 MCP 规范——缺少 structuredContent 字段,这会导致 Cursor 报错。解决办法是写一个 Python 包装脚本,在中间层添加缺失的字段。

graph LR
    subgraph Problem["❌ 问题"]
        P1["Cursor"] -->|"JSON-RPC"| P2["mcpbridge"]
        P2 -->|"响应缺少<br/>structuredContent"| P1
        P1 --> P3["⛔ 报错!"]
    end

    subgraph Solution["✅ 解决方案"]
        S1["Cursor"] -->|"JSON-RPC"| S2["🐍 Python Wrapper<br/>mcpbridge-wrapper"]
        S2 -->|"透传请求"| S3["mcpbridge"]
        S3 -->|"原始响应"| S2
        S2 -->|"注入<br/>structuredContent"| S1
        S1 --> S4["🎉 正常工作"]
    end

    style Problem fill:#ffebee,stroke:#c62828
    style Solution fill:#e8f5e9,stroke:#2e7d32
    style P3 fill:#ffcdd2,stroke:#c62828
    style S4 fill:#c8e6c9,stroke:#2e7d32

步骤 1:创建脚本 ~/bin/mcpbridge-wrapper(记得 chmod +x):

#!/usr/bin/env python3
"""
Wrapper for xcrun mcpbridge that adds structuredContent to responses.
"""
import sys, json, subprocess, threading

def process_response(line):
    try:
        data = json.loads(line)
        if isinstance(data, dict) and 'result' in data:
            result = data['result']
            if isinstance(result, dict):
                if 'content' in result and 'structuredContent' not in result:
                    content = result.get('content', [])
                    if isinstance(content, list) and len(content) > 0:
                        for item in content:
                            if isinstance(item, dict) and item.get('type') == 'text':
                                text = item.get('text', '')
                                try:
                                    result['structuredContent'] = json.loads(text)
                                except json.JSONDecodeError:
                                    result['structuredContent'] = {"text": text}
                                break
        return json.dumps(data)
    except json.JSONDecodeError:
        return line

def main():
    proc = subprocess.Popen(
        ['xcrun', 'mcpbridge'] + sys.argv[1:],
        stdin=subprocess.PIPE, stdout=subprocess.PIPE,
        stderr=sys.stderr, text=True, bufsize=1
    )

    def pipe_output(stdout):
        for line in stdout:
            print(process_response(line.strip()), flush=True)

    threading.Thread(target=pipe_output, args=(proc.stdout,), daemon=True).start()

    for line in sys.stdin:
        proc.stdin.write(line)
        proc.stdin.flush()

if __name__ == '__main__':
    main()

步骤 2:修改 ~/.cursor/mcp.json

{
  "mcpServers": {
    "xcode-tools": {
      "command": "/Users/YOUR_USERNAME/bin/mcpbridge-wrapper"
    }
  }
}

重启 Cursor,问题解决。


八、从 GCC 到 MCP:编译器与工具链的进化之路

写到这里,我突然想起多年前研究 Clang 和 Swift 编译时写的一篇文章(就是简书上那篇《OC 与 Swift 编译对比》)。当时梳理了从 GCC 到 LLVM 再到 Swift 的历史,现在回头看,MCP 的出现其实也是这条进化线上的必然一环。

timeline
    title 编译器与工具链进化史
    section GCC 时代
        2000s : GCC 作为 Xcode 默认编译器
              : 编译器是"黑盒"
              : IDE 交互能力有限
              : 扩展困难
    section LLVM/Clang 时代
        2007 : Chris Lattner 创建 LLVM
             : Clang 取代 GCC
             : 模块化设计 + libTooling
             : SourceKit 诞生
    section Swift 编译器
        2014 : Swift 语言发布
             : SIL 中间表示层
             : ARC 优化 / 泛型特化
             : 智能代码补全基础
    section MCP 时代
        2025-2026 : Xcode Intelligence
                  : mcpbridge MCP Server
                  : AI 可"操作"代码
                  : 编译-测试-预览全闭环

为什么这么说?

  • GCC 时代:编译器是个"黑盒",只负责把源码变成机器码,与 IDE 的交互很有限。Xcode 早期也是通过调用 GCC 来完成构建,但想要扩展功能(比如代码索引、实时诊断)非常困难。

  • LLVM/Clang 时代:LLVM 的模块化设计让编译器变成了可重用的库。Clang 提供了 libTooling,开发者可以编写插件遍历 AST,实现代码检查、重构。Xcode 的 SourceKit 也应运而生,为 IDE 提供了实时的代码分析能力。

  • Swift 编译器:更进一步,引入了 SIL (Swift Intermediate Language),在 LLVM IR 之前增加了一层高级中间表示,专门用于 Swift 特有的优化(如 ARC 优化、泛型特化)。这为更智能的代码补全和诊断打下了基础。

  • MCP 时代:现在,我们把编译器 + IDE 的能力通过 MCP 暴露给 AI。AI 不再是"看"代码,而是能"操作"代码——读取、修改、构建、测试、预览,整个闭环自动化。

graph BT
    subgraph Evolution["📈 能力进化路径"]
        direction BT
        L1["🔧 GCC<br/>编译源码 → 机器码<br/>黑盒,不可扩展"]
        L2["⚙️ LLVM / Clang<br/>模块化编译器库<br/>libTooling + AST 遍历<br/>SourceKit 实时分析"]
        L3["🦅 Swift Compiler<br/>SIL 中间表示<br/>ARC / 泛型优化<br/>智能补全基础"]
        L4["🤖 MCP<br/>编译器 + IDE 可编程化<br/>AI 直接操作工程<br/>读-写-编译-测试-预览 闭环"]

        L1 -->|"模块化突破"| L2
        L2 -->|"语言级创新"| L3
        L3 -->|"AI 可编程化"| L4
    end

    style L1 fill:#efebe9,stroke:#4e342e
    style L2 fill:#e3f2fd,stroke:#1565c0
    style L3 fill:#fff3e0,stroke:#e65100
    style L4 fill:#e8f5e9,stroke:#2e7d32,stroke-width:3px

从 GCC 到 LLVM,我们解决了编译器的模块化;从 SourceKit 到 MCP,我们解决了 IDE 能力的可编程化。每一步都在打破工具的边界,让开发者(或 AI)能更深入地控制开发环境。


九、总结与展望

Xcode MCP Server 的出现,意味着 AI 辅助编程进入了一个新阶段:从聊天式代码生成,进化到工程级自动运维。你可以让 AI 帮你修编译错误、跑测试、甚至验证 UI,而不再只是粘贴代码让你自己试错。

当然,目前还有些粗糙(比如 Cursor 兼容性问题),但方向已经非常明确。未来,随着 Apple Intelligence 的成熟和第三方模型的接入,Xcode 可能会变成一个"AI 优先"的 IDE——你只需要描述需求,剩下的交给 AI 和 MCP 去执行。

graph LR
    subgraph Past["📼 过去"]
        P1["AI 生成代码片段"] --> P2["复制粘贴到 IDE"] --> P3["手动编译调试"]
    end

    subgraph Present["📍 现在 (MCP)"]
        N1["AI 理解工程结构"] --> N2["直接读写文件"] --> N3["自动编译测试"]
    end

    subgraph Future["🔮 未来"]
        F1["描述需求"] --> F2["AI 自主规划"] --> F3["全自动交付"]
    end

    Past -.->|"进化"| Present
    Present -.->|"展望"| Future

    style Past fill:#efebe9,stroke:#795548
    style Present fill:#e8f5e9,stroke:#2e7d32,stroke-width:2px
    style Future fill:#e1f5fe,stroke:#0277bd

奇奇怪怪的无用知识又增加了:从 GCC 到 LLVM 再到 MCP,每一步都是因为开发者"受不了"现有工具的局限而推动的。历史总是惊人地相似,但每次进化都让工具离人更近一步。


❌