body: Self.Body 和 body: View的区别

问:能否将

public protocol View {
    // 注意:这并不是在定义一个Body变量。定义了一个关联类型,它表示某个遵守该协议的类型所必须提供的一种类型别名。
    associatedtype Body : View
    // Self.Body 指的是当前类型的 Body 类型,也就是遵守 View 协议的具体类型为它的 Body 所定义的类型。例如,如果有一个类型 MyView 遵守了 View 协议,那么 Self.Body 在这个上下文中就代表 MyView.Body
    @ViewBuilder @MainActor var body: Self.Body { get }
}

简化如下:

public protocol View {
    @ViewBuilder @MainActor var body: View { get }
}

答:不能

是的,按照你给出的简化定义:

public protocol View {
    @ViewBuilder @MainActor var body: View { get }
}

这个写法确实表示 body 必须返回一个遵守 View 协议的类型。然而,虽然这看起来是合理的,但它与原始定义中的 associatedtype 存在重要区别。我们深入探讨一下这种写法的具体含义和潜在问题:

1. 返回类型的限制

在这个定义中:

var body: View { get }

body 的返回类型被固定为 View 协议,即意味着任何遵循 View 协议的具体类型都可以作为 body 的返回值。但是,问题在于这里的 View 是一个协议类型,而不是一个具体的类型。这会导致 Swift 失去对具体视图类型的编译时优化。也就是说,body 返回的视图类型没有被具体化,所有 View 类型的对象都被视为一种抽象类型(存在性类型),这将引发性能问题。

在 Swift 中,协议类型(即 View 本身)与具体类型(如 Text 或 HStack)的区别在于协议类型需要更多的动态派发。协议类型在运行时确定其类型和行为,而具体类型在编译时就被完全解析和优化。

2. 与 some View 的对比

原始的 View 协议使用了 associatedtype Body: View 来允许 body 返回一个遵守 View 协议的具体类型。这个设计允许使用 some View 来返回一个具体的视图类型,而不仅仅是遵守 View 协议的任何视图。

例如,原始的 View 协议允许这样的写法:

struct MyView: View {
    var body: some View {
        Text("Hello, world!")
    }
}

这里的 some View 告诉编译器 body 返回的是一个具体的 View,并且编译器可以推断出具体的类型是 Text。这种方式允许编译时的类型推断和优化。

但是在你的简化定义中,body 的返回值被写死为 View,这意味着无法使用 some View,编译器也不能推断 body 返回的具体类型。这可能会导致 SwiftUI 中的视图组合失去编译时优化。

3. 编译时优化的损失

使用 associatedtype 和 some View 的原始设计允许 SwiftUI 在编译时对视图层次结构进行优化,因为每个视图的具体类型在编译时已经确定。通过这种优化,SwiftUI 能够生成高效的代码,无需在运行时进行复杂的类型检查。

但是,如果你把 body 的类型固定为 View(协议类型),Swift 就无法在编译时具体化视图的类型信息,这可能导致运行时性能下降,特别是在视图层次结构复杂的情况下。

4. 协议的动态派发问题

在 Swift 中,使用协议类型作为返回值(例如 View)意味着需要通过动态派发来处理类型。动态派发通常需要额外的开销,因为系统需要在运行时查找和解析具体的实现。这与 some View 的编译时静态分发相比,效率要低。

总结

虽然简化定义确实表示 body 返回的类型需要遵守 View 协议,但它缺乏原始 associatedtype Body: View 的灵活性和编译时优化。特别是无法使用 some View 和具体类型的推断,可能导致性能损失。因此,尽管从表面上看它更简单,但实际应用中它会影响 SwiftUI 的性能和编译时类型安全。

因此,原始的写法:

associatedtype Body: View
保留了更多灵活性,允许具体类型推断和优化,而你的简化写法在性能和优化上有所妥协。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,324评论 5 476
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,303评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,192评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,555评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,569评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,566评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,927评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,583评论 0 257
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,827评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,590评论 2 320
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,669评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,365评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,941评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,928评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,159评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,880评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,399评论 2 342

推荐阅读更多精彩内容