为什么 LocalizedStringKey 的行为取决于我是否将字符串插值传递给其初始化程序?

2023-12-12

在试图回答的同时这个问题,我发现了一个奇怪的行为。

Text(LocalizedStringKey("Hello \(Image(systemName: "globe"))"))

显示一个地球仪,但是

Text(LocalizedStringKey("Hello {world}".replacingOccurrences(of: "{world}", with: "\(Image(systemName: "globe"))")))
Text(LocalizedStringKey("Hello" + "\(Image(systemName: "globe"))"))

显示“Hello”,然后是一堆 SwiftUI 内部术语。

一个更简单的例子是:

let x = "\(Image(systemName: "globe"))"
print(LocalizedStringKey.init(x))
print(LocalizedStringKey.init("\(Image(systemName: "globe"))"))

我传递给的值LocalizedStringKey.init应该是相同的,两者"\(Image(systemName: "globe"))",但是第一个打印

LocalizedStringKey(key: "%@", hasFormatting: true, arguments: [...])

和第二张照片

LocalizedStringKey(key: "Image(provider: SwiftUI.ImageProviderBox<SwiftUI.Image.(unknown context at $7ff91ccb3380).NamedImageProvider>)", hasFormatting: false, arguments: [])

看起来LocalizedStringKey.init更改其行为取决于我传递的参数是否是(内插的)字符串文字。

据我所知,这两个电话LocalizedStringKey.init正在调用相同的初始化程序。只有一个无参数标签初始化程序 in LocalizedStringKey,这需要一个String.

如果还有一个初始化器需要LocalizedStringKey,结果会更容易理解。LocalizedStringKey 具有自定义字符串插值规则, and 一个专门针对Image, 毕竟。但据我所知,这是唯一没有参数标签的初始化程序。

如果初始化器的参数是@autoclosure () -> String。如果我传入的表达式被延迟求值,则该方法可能能够通过某种我未知的方式“窥视”闭包内部。但该参数不是自动关闭。

这里似乎发生的是编译器正在创建一个LocalizedStringKey with key与您传入的插值具有相同的模式,即使参数是 String!

这里到底发生了什么?我是否在某处错过了隐藏的初始化程序?


TL;DR:您所看到的行为来自ExpressibleByStringInterpolation。但请继续阅读以获得更多乐趣!

LocalizedStringKey如果您纯粹将其视为一种方便,允许 SwiftUI 界面元素在使用字符串文字时“免费”本地化,那么就更容易理解了。只有一个实时您可以直接使用它。

考虑Text。有两个相关的初始化器:

init(_ key: LocalizedStringKey, tableName: String? = nil, bundle: Bundle? = nil, comment: StaticString? = nil)

它将尝试本地化传入的文本,并且

init<S>(_ content: S) where S : StringProtocol

这将显示字符串而不改变它。

如果你打电话Text("Hello"),使用哪个初始化器?

字符串文字符合StringProtocol, but LocalizedStringKey也是ExpressibleByStringLiteral。编译器不知道选择哪一个。

为了获得“免费”本地化,StringProtocol初始化器标记为@_disfavoredOverload,它告诉编译器假设字符串文字是LocalizableStringKey而不是一个String.

所以,Text("Hello") and Text(LocalizedStringKey("Hello"))是等价的。

let string = "Hello"
Text(string)

在这种情况下,不存在冲突 - 编译器使用StringProtocol初始值设定项和字符串未本地化。

这和你的问题有什么关系?LocalizedStringKey is also ExpressibleByStringInterpolation,这就是“隐藏初始化程序”的来源。但与上面的示例一样,只有当您使用单个内插字符串对其进行初始化时,这才会发挥作用。

Text("Hello \(Image(systemName: "globe"))")

您正在传递一个插值字符串,因此编译器可以处理它并将图像添加到插值中。

Text("Hello {world}".replacingOccurrences(of: "{world}", with: "\(Image(systemName: "globe"))"))

Here, replacingOccurrences(of:首先被评估,这意味着你的论点是String,它不被视为通过字符串插值表示的 LocalizedStringKey。您实质上看到的是图像的描述。

类似的情况也发生在以下示例中+在里面。这隐含地使得String,所以你失去了特殊的图像插值LocalizedStringKey给你。

对于您的最后一个代码示例:

let x = "\(Image(systemName: "globe"))"
print(LocalizedStringKey.init(x))
print(LocalizedStringKey.init("\(Image(systemName: "globe"))"))

x是包含图像描述的字符串。记住,只LocalizedStringKey具有真正理解和表达的魔力Image。任何其他字符串插值都将回退到插值对象的描述。

第一个初始化器传递一个字符串(它被视为一个键,这是您真正直接使用的唯一一次LocalizedStringKey,如果您在运行时生成键并希望使用它们进行查找)。

第二个初始化器正在使用ExpressibleByStringInterpolation并正在使用LocalizedStringKey.StringInterpolation将图像插入到其内部存储中,然后可以通过以下方式进行渲染Text.

本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

为什么 LocalizedStringKey 的行为取决于我是否将字符串插值传递给其初始化程序? 的相关文章

随机推荐