【问题标题】:Default property value with closure makes a compiler to recompile all files带闭包的默认属性值使编译器重新编译所有文件
【发布时间】:2026-01-11 19:15:02
【问题描述】:

这个source 有一段使用闭包或函数设置默认属性值,我们可以在其中找到示例

以下是如何使用闭包提供默认属性值的概要:

class SomeClass {

    let someProperty: SomeType = {
         // create a default value for someProperty inside this closure
         // someValue must be of the same type as SomeType
         return someValue
    }() 
}

嗯,我经常使用它...而且,我经常只更改一个符号后等待整个项目重新编译。而今天我发现这两件事是相互关联的。

假设我们有一个类,我们在其中设置一些带有闭包和函数的默认属性

class Class1 {
    
    let value: Int
    
    init(_ value: Int) {
        self.value = value
    }
    
    private lazy var lazyValueWithClosure: Int = {
        return 1111
    }()
    
    private lazy var lazyValueWithFunction: Int = self.getValue()
    
    private func getValue() -> Int {
        return 2222
    }
}

我们在一个单独的文件中还有一些其他类,我们使用上面的Class1

class Class2 {
    
    let value: Int
    
    init(_ value: Int) {
        self.value = value
        _ = Class1(100)
    }
}

以及我们使用Class2的单独文件中的其他一些类

class Class3 {
    
    let value: Int
    
    init(_ value: Int) {
        self.value = value
        _ = Class2(100)
    }
}

等等……

我决定使用terminal + xcodebuild + grep 仅获取有关重新编译文件的信息。这是我用来获取编译信息的命令:

xcodebuild -scheme Test -sdk iphonesimulator -arch x86_64 -configuration Debug build OTHER_SWIFT_FLAGS="-Xfrontend -debug-time-function-bodies" | grep '^[0-9]\{1,20\}.[0-9]\{1,20\}ms.*init(_ value: Int)'

这就是准备工作。现在我们转到Class1 并将2222 更改为其他值。运行上面的命令,得到结果。

0.1ms   /Users/iwheelbuy/Documents/recompile/Test/Classes/Class1.swift:11:5 init(_ value: Int)

结果很好。使用函数设置默认值按预期工作。我们更改了一个文件,只编译了一个文件。

然后让我们将值1111Class1 更改为其他值并运行命令。终端输出现在如下所示:

0.8ms   /Users/iwheelbuy/Documents/recompile/Test/Classes/Class5.swift:11:5 init(_ value: Int)
0.3ms   /Users/iwheelbuy/Documents/recompile/Test/Classes/Class1.swift:11:5 init(_ value: Int)
1.0ms   /Users/iwheelbuy/Documents/recompile/Test/Classes/Class4.swift:11:5 init(_ value: Int)
0.3ms   /Users/iwheelbuy/Documents/recompile/Test/Classes/Class3.swift:11:5 init(_ value: Int)
0.3ms   /Users/iwheelbuy/Documents/recompile/Test/Classes/Class2.swift:11:5 init(_ value: Int)

所有类都重新编译...现在假设您有一个大型项目,并且默认值闭包中的任何微小更改都会让您等待整个项目重新编译。

问题:

  • 是什么原因?
  • 关于如何使用默认值闭包且避免重新编译的任何建议?
  • this topic相关?

【问题讨论】:

    标签: swift xcode terminal swift3 xcodebuild


    【解决方案1】:

    这是 Swift 编译器中的一个已知问题。问题是一旦你使用像这样的闭包或惰性属性,每个 Swift 文件都会被类型检查。我写了一篇关于这个主题的博文,你可以找到here

    【讨论】: