- r - 以节省内存的方式增长 data.frame
- ruby-on-rails - ruby/ruby on rails 内存泄漏检测
- android - 无法解析导入android.support.v7.app
- UNIX 域套接字与共享内存(映射文件)
我们正试图弄清楚这是 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 1
和OPTION 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 是如何开始解决这个问题的。嗯,someNumber
是Double?
, 所以它试图把它变成:
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/
在检查对象是否为 nil 时,有人使用 1: if (object == nil) { //... } 有人用 2: if (nil == object) { //... } 1和2有
func InsertApData(w http.ResponseWriter, r *http.Request) { decoder := json.NewDecoder(r.Body)
为什么 nil.to_s 返回 "",而 nil.inspect 返回 "nil"(当显然 .inspect 使用 .to_s 方法) 最佳答案 "inspect" method of the Obj
运行时间 https://play.golang.org/p/sl12vfS9vP package main import "fmt" func main() { err := run()
我可以将三元条件运算符用于 if {} else {} 语句,如下所示:a ? x : y,还是问题?回答 1 : 回答 2。 是否可以使用这种格式来检查,而不是 a 是 true 还是 false,
无法弄清楚这里出了什么问题。按照设计设置说明,用谷歌搜索我能想到的一切,仍然没有运气。 undefined method `email' for # Extracted source (around
今天写一些 rspec 时,我遇到了一些意外的行为,将日期(和时间)实例与 nil 进行比较。这是一个使用原始 ruby 的示例(没有 Rails 或其他库): user@MacBook-Work
我将数据类型的非零值分配给非可选属性,然后将其分配给可选属性,最后用所述数据实例化图像。当可选项通过 if-let 子句时,它的 block 执行,抛出错误: Fatal error: Unexpec
swift 5.1 。 考虑以下。 let x: Any? = nil let y: Any = x print("x \(x)") // x nil pri
请耐心听我解释这一点, 我正在创建一个聊天室,用户可以在其中上传照片供其他人查看。当用户点击图标时,他们可以将照片上传到我的 Firebase 数据库(这已成功完成,我已经对此进行了测试)。照片的 U
我的 xCode 5.0 目前遇到一个奇怪的问题:一个对象在控制台中似乎为 nil,但同时它可以通过代码访问。 图 1:对象似乎为零 图2:NSLog(@"%@", imgDownloader) 的输
我有一个实现协议(protocol)的类,以便添加 3 个变量。我专门设置了图像变量,调试器显示该变量存在,但是在我打印它时在代码中显示为 nil,我的 if let 语句也认为该变量为 nil。 @
这个问题在这里已经有了答案: Swift 2 ( executeFetchRequest ) : error handling (5 个答案) 关闭 7 年前。 开始在 SWIFT 中学习编码,每次
两者 (not 'nil) 和 (not nil) 求值为T,那么'nil和nil有什么区别吗?那么 ''nil 呢?如果 ''nil 的计算结果为 'nil,那么 ''nil 是否也应该计算为 ni
(if '(nil nil) 'print-true 'print-false) (if '(nil) 'print-true 'print-false) 在上面的代码
这个问题在这里已经有了答案: 关闭 11 年前。 Possible Duplicate: obj.nil? vs. obj == nil 我发现了一个问题 - == nil 和 nil 哪个更好?
此代码段抛出异常: x = nil jsoned = x.to_json puts 'x.to_json=' + jsoned.inspect puts 'back=' + JSON.parse(js
基本上我有一个对象想要传递给前端。我在后端记录了它,它不是空的,但是在前端,当我提醒它时,它变成了空。 ... presentation := &presentationStruct { Obje
我创建了一个自定义错误类型来包装错误,以便在 Golang 中更轻松地进行调试。当有错误要打印时它可以工作,但现在它会引起 panic 。 Demo type Error struct { E
写一个符合我当前问题的标题有点难..我有一个 main() 函数,它使用另一个包 (database_sql) 中的函数。该函数初始化一个全局变量 sql.DB*。初始化后,变量不为nil,但是对于其
我是一名优秀的程序员,十分优秀!