gpt4 book ai didi

swift - 将 nil-coalescing 运算符与两个可选值一起使用时类型推断失败

转载 作者:IT王子 更新时间:2023-10-29 05:51:03 24 4
gpt4 key购买 nike

我们正试图弄清楚这是 Swift 中的错误还是我们滥用了泛型、可选类型、类型推断和/或 nil 合并运算符。

我们的框架包含一些用于将字典解析为模型的代码,我们遇到了具有默认值的可选属性的问题。

我们有一个协议(protocol) SomeProtocol 和两个在协议(protocol)扩展中定义的通用函数:

mapped<T>(...) -> T?
mapped<T : SomeProtocol>(...) -> T?

我们的结构和类遵循该协议(protocol),然后在协议(protocol)要求的初始化函数中解析它们的属性。

init(...) 函数中,我们尝试像这样设置属性 someNumber 的值:

someNumber = self.mapped(dictionary, key: "someNumber") ?? someNumber

字典当然包含键 someNumber 的实际值。但是,这总是会失败,并且永远不会从 mapped() 函数返回实际值。

注释掉第二个泛型函数或强制向下转换赋值右侧的值将解决此问题,但我们认为这应该按照当前编写的方式工作。


下面是演示该问题的完整代码片段,以及(暂时)修复代码中标记为OPTION 1OPTION 2 的问题的两个选项:

import Foundation

// Some protocol

protocol SomeProtocol {
init(dictionary: NSDictionary?)
}

extension SomeProtocol {
func mapped<T>(dictionary: NSDictionary?, key: String) -> T? {
guard let dictionary = dictionary else {
return nil
}

let source = dictionary[key]
switch source {

case is T:
return source as? T

default:
break
}

return nil
}

// ---
// OPTION 1: Commenting out this makes it work
// ---

func mapped<T where T:SomeProtocol>(dictionary: NSDictionary?, key: String) -> T? {
return nil
}
}

// Some struct

struct SomeStruct {
var someNumber: Double? = 0.0
}

extension SomeStruct: SomeProtocol {
init(dictionary: NSDictionary?) {
someNumber = self.mapped(dictionary, key: "someNumber") ?? someNumber

// OPTION 2: Writing this makes it work
// someNumber = self.mapped(dictionary, key: "someNumber") ?? someNumber!
}
}

// Test code

let test = SomeStruct(dictionary: NSDictionary(object: 1234.4567, forKey: "someNumber"))
if test.someNumber == 1234.4567 {
print("success \(test.someNumber!)")
} else {
print("failure \(test.someNumber)")
}

请注意,这是一个缺少 mapped 函数实际实现的示例,但结果是相同的,对于这个问题,代码应该足够了。


编辑:我之前报告过这个问题,现在它被标记为已修复,所以希望这不会再发生在 Swift 3 中。
https://bugs.swift.org/browse/SR-574

最佳答案

您为编译器提供了太多选项,而它选择了错误的选项(至少不是您想要的选项)。问题是每个 T可以简单地提升到 T? , 包括 T? (提升到 T?? )。

someNumber = self.mapped(dictionary, key: "someNumber") ?? someNumber

哇。这样的类型。所以可选。 :D

那么 Swift 是如何开始解决这个问题的。嗯,someNumberDouble? , 所以它试图把它变成:

Double? = Double?? ?? Double?

那有用吗?让我们寻找一个通用的 mapped ,从最具体的开始。

func mapped<T where T:SomeProtocol>(dictionary: NSDictionary?, key: String) -> T? {

为了使这项工作成功,T必须是 Double? .是Double?:SomeProtocol ?没有。继续前进。

func mapped<T>(dictionary: NSDictionary?, key: String) -> T? {

这个有用吗?当然! T可以是Double?我们返回Double??一切都解决了。

那么为什么这个有效呢?

someNumber = self.mapped(dictionary, key: "someNumber") ?? someNumber!

这解析为:

Double? = Optional(Double? ?? Double)

然后事情按照您认为应该的方式运作。

小心这么多可选。是否someNumber真的必须是可选的吗?这些东西都应该throw ? (我并不是说 throw 是可选问题的一般解决方法,但至少这个问题让您有时间考虑这是否真的是错误情况。)

在 Swift 中以返回值类型参数化几乎总是一个坏主意 mapped做。这在 Swift 中往往是一个真正的困惑(或任何具有大量类型推断的通用语言,但当涉及 Optionals 时它在 Swift 中真的会爆炸)。类型参数通常应出现在参数中。如果您尝试类似的操作,您会发现问题所在:

let x = test.mapped(...)

它无法推断出 x 的类型.这不是反模式,有时麻烦是值得的(公平地说,您正在解决的问题可能是其中一种情况),但如果可以的话请避免它。

但杀死你的是可选项。


编辑:Dominik 提出了一个很好的问题,即为什么当 mapped 的受限版本时它的行为会有所不同。已移除。我不知道。显然,类型匹配引擎检查有效类型的顺序略有不同,具体取决于有多少种方式 mapped是通用的。您可以通过添加 print(T.self) 来查看至 mapped<T> .这可能被认为是编译器中的错误。

关于swift - 将 nil-coalescing 运算符与两个可选值一起使用时类型推断失败,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/34856645/

24 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com