Swift中有两种不同类型的nil?

Mic*_*ael 13 swift

我在使用REPL(= Read-Eval-Print-Loop)时在Swift语言中遇到了一些奇怪的行为,其中似乎有两种不同类型的nil在运行时具有不同的行为:

为此,我定义了一个函数g:

func g(x:String!) {
    println("start!");
    println((x == "foo") ? "foo" : "not");
}
Run Code Online (Sandbox Code Playgroud)

然后我定义了两个变量:

var x:String
var y:String!
Run Code Online (Sandbox Code Playgroud)

当我打电话时g(x),它就像Objective-C一样:

start!
not
Run Code Online (Sandbox Code Playgroud)

当我打电话时g(y),我收到一个错误:

start!
fatal error: Can't unwrap Optional.None
Execution interrupted. Enter Swift code to recover and continue.
Enter LLDB commands to investigate (type :help for assistance.)
Run Code Online (Sandbox Code Playgroud)

请注意,此错误在运行时被捕获!该功能已经开始.你可以看到这个因为"开始!" 输出中的字符串.

看来,在第一种情况下,函数的值为nil,并附有"这不是零值"的注释.在第二种情况下,似乎得到一个零,附加说明"这个值是零,请尖叫,不要只使用它."

为什么我在第一种情况下没有出错,在第二种情况下出错?不应该是g(x)和g(y)的行为相同吗?

这是预期的行为吗?我错过了什么吗?这是Swift中的错误吗?规范中的错误?或者实施中的错误?或者只是REPL中的一个错误?是不是不可能访问未初始化的变量?


整个会议记录,供参考......

$ /Applications/Xcode6-Beta.app/Contents/Developer/Toolchains/XcodeDefault.xctoolchain/usr/bin/swift
Welcome to Swift!  Type :help for assistance.
  1> func g(x:String!) {println("start!"); println((x=="foo") ? "foo" : "not");} 
  2> var x:String
x: String = {
  core = {
    _baseAddress = Builtin.RawPointer = 0x0000000000000000
    _countAndFlags = 0
    _owner = None
  }
}
  3> var y:String!
y: String! = nil
  4> g(x)
start!
not
  5> g(y)
start!
fatal error: Can't unwrap Optional.None
Execution interrupted. Enter Swift code to recover and continue.
Enter LLDB commands to investigate (type :help for assistance.)
  6> 
Run Code Online (Sandbox Code Playgroud)

要重现整个会话,只需在终端中键入以下代码:

/Applications/Xcode6-Beta.app/Contents/Developer/Toolchains/XcodeDefault.xctoolchain/usr/bin/swift
func g(x:String!) {println("start!"); println((x=="foo") ? "foo" : "not");}
var x:String
var y:String!
g(x)
g(y)
Run Code Online (Sandbox Code Playgroud)

(该函数写在一行,因为复制和粘贴功能在REPL中被破坏,至少在我的机器上.但是在一行中编写它是有效的.这是另一个故事...)


2015年3月更新: Apple似乎解决了这个问题.如果没有初始值,则无法再声明正常变量:

 2> var x:String
repl.swift:2:1: error: variables currently must have an initial value when entered at the top level of the REPL
var x:String
^
Run Code Online (Sandbox Code Playgroud)

Bry*_*hen 14

REPL会自动为您做一些初始化,即使您不想要它

我将您的代码复制并粘贴到操场上,它会向我抛出此错误:

error: variable 'x' used before being initialized
g(x)
  ^
<REPL>:22:5: note: variable defined here
var x:String
Run Code Online (Sandbox Code Playgroud)

我认为这是正确的结果.代码不应该编译.

当我输入REPL时

var a:String
Run Code Online (Sandbox Code Playgroud)

它打印

a: String = {
  core = {
    _baseAddress = Builtin.RawPointer = 0x0000000000000000
    _countAndFlags = 0
    _owner = None
  }
}
Run Code Online (Sandbox Code Playgroud)

所以REPL以某种方式初始化a为一个nil字符串(我不确定它是否合法存在),故事的其余部分在其他答案中解释


看起来REPL正在为每个变量进行一些默认初始化,因此您无法使用未初始化的变量

Welcome to Swift!  Type :help for assistance.
  1> var a:String
a: String = {
  core = {
    _baseAddress = Builtin.RawPointer = 0x0000000000000000
    _countAndFlags = 0
    _owner = None
  }
}
  2> var b:Int
b: Int = 0
  3> var c:Int[]
c: Int[] = size=0
  4> var d:Dictionary<Int,Int>
d: Dictionary<Int, Int> = {}
Run Code Online (Sandbox Code Playgroud)

更有趣的是,它仍然可以初始化非可选类型 nil

  5> import Foundation
  6> var f:NSObject
f: NSObject = {}
  7> var g:NSNumber
g: NSNumber = {
  Foundation.NSValue = <parent is NULL>

}
  8> print(g)
fatal error: Can't unwrap Optional.None
Execution interrupted. Enter Swift code to recover and continue.
Run Code Online (Sandbox Code Playgroud)

因此,REPL将对未初始化的变量(应该是编译时错误)的访问转变为运行时错误

Welcome to Swift!  Type :help for assistance.
  1> class Test{ var val:Int; init(v:Int) {val=v} }
  2> var t:Test
t: Test = {
  val = <parent is NULL>

}
  3> t.val
Execution interrupted. Enter Swift code to recover and continue.
Enter LLDB commands to investigate (type :help for assistance.)
  4> t = Test(v:1)
  5> t.val
$R2: Int = 1
  6> 
Run Code Online (Sandbox Code Playgroud)