是否有一种方法可以在swift中打印变量的运行时类型?例如:
var now = NSDate()
var soon = now.dateByAddingTimeInterval(5.0)
println("\(now.dynamicType)")
// Prints "(Metatype)"
println("\(now.dynamicType.description()")
// Prints "__NSDate" since objective-c Class objects have a "description" selector
println("\(soon.dynamicType.description()")
// Compile-time error since ImplicitlyUnwrappedOptional<NSDate> has no "description" method
在上面的例子中,我正在寻找一种方法来显示变量“soon”的类型是ImplicitlyUnwrappedOptional<NSDate>,或至少NSDate!
我目前的Xcode版本是6.0 (6A280e)。
import Foundation
class Person { var name: String; init(name: String) { self.name = name }}
class Patient: Person {}
class Doctor: Person {}
var variables:[Any] = [
5,
7.5,
true,
"maple",
Person(name:"Sarah"),
Patient(name:"Pat"),
Doctor(name:"Sandy")
]
for variable in variables {
let typeLongName = _stdlib_getDemangledTypeName(variable)
let tokens = split(typeLongName, { $0 == "." })
if let typeName = tokens.last {
println("Variable \(variable) is of Type \(typeName).")
}
}
输出:
Variable 5 is of Type Int.
Variable 7.5 is of Type Double.
Variable true is of Type Bool.
Variable maple is of Type String.
Variable Swift001.Person is of Type Person.
Variable Swift001.Patient is of Type Patient.
Variable Swift001.Doctor is of Type Doctor.
当使用Cocoa(而不是CocoaTouch)时,你可以为NSObject的子类对象使用className属性。
println(now.className)
这个属性对于普通的Swift对象是不可用的,它不是NSObject的子类(事实上,在Swift中没有根id或对象类型)。
class Person {
var name: String?
}
var p = Person()
println(person.className) // <- Compiler error
在CocoaTouch中,此时没有办法获得给定变量类型的字符串描述。类似的功能在Cocoa或CocoaTouch中也不存在。
Swift REPL能够打印出值的摘要,包括它的类型,所以这种内省方式可能在未来通过API实现。
编辑:转储(对象)似乎做的把戏。
您仍然可以通过className(返回String)访问该类。
实际上有几种方法来获取类,例如classForArchiver, classForCoder, classForKeyedArchiver(都返回AnyClass!)
无法获取原语的类型(原语不是类)。
例子:
var ivar = [:]
ivar.className // __NSDictionaryI
var i = 1
i.className // error: 'Int' does not have a member named 'className'
如果你想获取一个原语的类型,你必须使用bridgetooobjecvec()。例子:
var i = 1
i.bridgeToObjectiveC().className // __NSCFNumber
我为自己开发的类(或您可以访问的类)找到了一个解决方案。
在你的对象类定义中放置以下计算属性:
var className: String? {
return __FILE__.lastPathComponent.stringByDeletingPathExtension
}
现在你可以像这样简单地调用对象的类名:
myObject.className
请注意,只有当类定义是在与您想要的类名称完全相同的文件中创建时,这才会起作用。
因为这是常见的情况,上面的答案应该做的大多数情况下。但在某些特殊情况下,你可能需要想出一个不同的解决方案。
如果你需要类(文件)本身的类名,你可以简单地使用这一行:
let className = __FILE__.lastPathComponent.stringByDeletingPathExtension
也许这个方法能帮助到一些人。