swift - Swift 的异构值类型数组是如何工作的?

标签 swift interface value-type

我是一名刚开始使用 Swift 的 C++ 程序员。我看了 Dave Abrahams 的 WWCD 演讲 "Protocol Orientated Programming in Swift"我对创建受协议(protocol)约束的异构值类型数组的方式很感兴趣。

要使用视频中的示例,给定协议(protocol) Drawable 和实现它的两个结构:

protocol Drawable {
    func draw(renderer: Renderer) // Renderer is another protocol
}

struct Circle : Drawable {
    func draw(renderer: Renderer) {
        // Implementation
    }
}

struct Rectangle : Drawable {
    func draw(renderer: Renderer) {
        // Implementation
    }
}

可以将 Drawable 定义为包含 Drawable 的数组

struct Diagram : Drawable {
    var elements: [Drawable] = []

    func draw(renderer: Renderer) {
        for e in elements {
            e.draw(renderer);
        }
    }
}

我的问题是,这个异构 elements 数组到底是如何工作的?由于 Drawable 的各种实现的大小可能不同,我看不出如何将它们布置在内存中的高效数组中。这是否意味着这样的“协议(protocol)数组”实际上是在表面下使用按元素堆分配和动态/虚拟函数调用?

最佳答案

我对此很好奇,虽然我没有足够的时间来彻底弄清楚它。我仍然认为我已经得到了一些近似值,可以将其作为答案放在这里。

首先,这是来自 Jason Bell 的 article,它提供了一些关于它在幕后如何工作的提示(不仅适用于 Swift,也适用于 Objective-C 和其他语言)。

其次,如果我采用这个简单的程序:

protocol Foo { }

struct Bar: Foo { }

var fooArray = [Foo]()

fooArray.append(Bar())
fooArray.append(Bar())
fooArray.append(Bar())

let arrayElement = fooArray[0]

print(arrayElement)

...并通过执行 LLVM IR 将其编译为 swiftc -emit-ir unveil.swift > unveil.ir 然后我可以找出以下对应于简单 IRfooArray.append(Bar()) 代码:

%15 = getelementptr inbounds %P6unveil3Foo_* %3, i32 0, i32 1
store %swift.type* bitcast (i64* getelementptr inbounds ({ i8**, i64, { i64, i8*, i32, i32, i8*, %swift.type** (%swift.type*)*, %swift.type_pattern*, i32, i32, i32 }*, %swift.type* }* @_TMfV6unveil3Bar, i32 0, i32 1) to %swift.type*), %swift.type** %15, align 8
%16 = getelementptr inbounds %P6unveil3Foo_* %3, i32 0, i32 2
store i8** getelementptr inbounds ([0 x i8*]* @_TWPV6unveil3BarS_3FooS_, i32 0, i32 0), i8*** %16, align 8
%17 = getelementptr inbounds %P6unveil3Foo_* %3, i32 0, i32 0
call void @_TFV6unveil3BarCfMS0_FT_S0_()
%18 = bitcast %P6unveil3Foo_* %3 to %swift.opaque*
call void @_TFSa6appendurfRGSaq__Fq_T_(%swift.opaque* noalias nocapture %18, %swift.type* %14, %Sa* nocapture dereferenceable(8) @_Tv6unveil8fooArrayGSaPS_3Foo__)

Here 你可以找到 LLVM IR 语法,但对我来说上面的意思是 Swift 数组实际上是指针数组。

此外,与 IR 类似,我可以进入相同 Swift 行的程序集,即:

leaq    __TWPV6unveil3BarS_3FooS_(%rip), %rax
leaq    __TMfV6unveil3Bar(%rip), %rcx
addq    $8, %rcx
movq    %rcx, -56(%rbp)
movq    %rax, -48(%rbp)
callq   __TFV6unveil3BarCfMS0_FT_S0_
leaq    __Tv6unveil8fooArrayGSaPS_3Foo__(%rip), %rdx
leaq    -80(%rbp), %rax
movq    %rax, %rdi
movq    -160(%rbp), %rsi
callq   __TFSa6appendurfRGSaq__Fq_T_

...再次,上面操纵指针,从而证实了理论。

最后,在 SILWitnessTable.h 处发现来自 swift.org 的 SIL header SILWitnessVisitor.hswift/include/swift/SIL/ 表明相同。

实际上,我猜测(我希望真正知道他在说什么的人在这里权衡)值类型(例如 struct s)和引用类型(读取 class es)在 Swift 的引擎盖下并没有太大的不同。可能主要区别在于是否强制执行写时复制。

关于swift - Swift 的异构值类型数组是如何工作的?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/34350113/

相关文章:

oop - 接口(interface)和抽象类有什么区别?

c# - 为什么结构中的引用类型表现得像值类型?

ios - 如何将日期转换为\/Date(1440156888750-0700)\/到 Swift 可以处理的内容?

arrays - 从 UIImage 数组获取值并转换为字符串

ios - 如何在展开之前分配这些对象?

javascript - ReactJS中ClassicComponentClass和ComponentClass的区别

ios - swift:未检测到 Sprite 套件碰撞和位掩码

c# - 为什么实现中允许的默认值与接口(interface)上定义的默认值不同?

c# - 盒装值类型比较

arrays - 如果数组是值类型并因此被复制,那么它们为什么不是线程安全的呢?