作者热门文章
- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
2015 年 8 月 28 日更新:
这将在 Swift 2 中解决
见 Twitter response from Swift compiler developer
2015 年 10 月 23 日更新:
使用 Swift 2 泛型,您仍然无法获得 rawValue。你可以得到相关的值。
原问题:
我有一些 generic reflection code写得很快。在该代码中,我无法获取基于枚举的属性的值。问题归结为我无法执行 .rawValue
在类型为 Any
的属性值上. Swift 反射代码将返回枚举类型的值 Any
.那么如何从 Any 到 AnyObject,它是枚举的 rawValue。
到目前为止,我发现的唯一解决方法是使用协议(protocol)扩展所有枚举。您可以在下面看到使用此变通方法没问题的单元测试。
有没有办法在不向原始枚举中添加代码的情况下解决这个问题?
对于我的反射代码,我需要 getRawValue
方法签名保持原样。
class WorkaroundsTests: XCTestCase {
func testEnumToRaw() {
let test1 = getRawValue(MyEnumOne.OK)
XCTAssertTrue(test1 == "OK", "Could nog get the rawvalue using a generic function")
let test2 = getRawValue(MyEnumTwo.OK)
XCTAssertTrue(test2 == "1", "Could nog get the rawvalue using a generic function")
let test3 = getRawValue(MyEnumThree.OK)
XCTAssertTrue(test3 == "1", "Could nog get the rawvalue using a generic function")
}
enum MyEnumOne: String, EVRawString {
case NotOK = "NotOK"
case OK = "OK"
}
enum MyEnumTwo: Int, EVRawInt {
case NotOK = 0
case OK = 1
}
enum MyEnumThree: Int64, EVRaw {
case NotOK = 0
case OK = 1
var anyRawValue: AnyObject { get { return String(self.rawValue) }}
}
func getRawValue(theEnum: Any) -> String {
// What can we get using reflection:
let mirror = reflect(theEnum)
if mirror.disposition == .Aggregate {
print("Disposition is .Aggregate\n")
// OK, and now?
// Thees do not complile:
//return enumRawValue(rawValue: theEnum)
//return enumRawValue2(theEnum )
if let value = theEnum as? EVRawString {
return value.rawValue
}
if let value = theEnum as? EVRawInt {
return String(value.rawValue)
}
}
var valueType:Any.Type = mirror.valueType
print("valueType = \(valueType)\n")
// No help from these:
//var value = mirror.value --> is just theEnum itself
//var objectIdentifier = mirror.objectIdentifier --> nil
//var count = mirror.count --> 0
//var summary:String = mirror.summary --> "(Enum Value)"
//var quickLookObject = mirror.quickLookObject --> nil
let toString:String = "\(theEnum)"
print("\(toString)\n")
return toString
}
func enumRawValue<E: RawRepresentable>(rawValue: E.RawValue) -> String {
let value = E(rawValue: rawValue)?.rawValue
return "\(value)"
}
func enumRawValue2<T:RawRepresentable>(rawValue: T) -> String {
return "\(rawValue.rawValue)"
}
}
public protocol EVRawInt {
var rawValue: Int { get }
}
public protocol EVRawString {
var rawValue: String { get }
}
public protocol EVRaw {
var anyRawValue: AnyObject { get }
}
最佳答案
不幸的是,目前这在 Swift 中看起来不太可能,但是我已经考虑了您的问题一段时间,我将提出 Swift 团队可以帮助您解决此问题的 3 种方法。
Any
value 以查看它是否是枚举,如果是,您想查看它是否具有原始值。 rawValue
属性应该可以通过以下代码访问:let mirror = reflect(theEnum) // theEnum is of Any type
for i in 0..<mirror.count {
if mirror[i].0 == "rawValue" {
switch mirror[i].1.value {
case let s as String:
return s
case let csc as CustomStringConvertible:
return csc.description
default:
return nil
}
}
}
count
的
0
.我真的认为这是 Swift 团队在实现
Swift._EnumMirror
时的疏忽。 ,我将就此事提交雷达报告。
rawValue
绝对是合法属性(property)。这是一个奇怪的场景,因为枚举不允许具有其他存储属性。此外,您的枚举声明从未明确符合
RawRepresentable
,也不声明
rawValue
属性(property)。编译器只会在您输入
enum MyEnum: String
时推断出或
: Int
或者什么。在我的测试中,似乎属性是在协议(protocol)中定义还是关联类型的实例无关紧要,它仍然应该是镜像中表示的属性。
RawRepresentable
因为 Swift 不知道 rawValue
是什么类型的属性(property)会回来。语法如 RawRepresentable<where RawValue == String>
是可以想象的,或者也许 protocol<RawRepresentable where RawValue == String>
.如果这是可能的,您可以尝试通过 switch 语句转换为类型,如下所示:switch theEnum {
case let rawEnum as protocol<RawRepresentable where RawValue == String>:
return rawEnum.rawValue
// And so on
}
RawRepresentable
,Swift 编译器告诉您只能在泛型函数中使用它,但是当我查看您的代码时,这只会让您陷入困境。泛型函数在编译时需要类型信息才能工作,而这正是您无法使用的
Any
实例。
MyGenericStruct<MyType>
和
MyGenericClass<MyType>
是合法专用的具体类型,您可以对其进行运行时检查并强制转换为。然而,Swift 团队可能有充分的理由不想用协议(protocol)来做到这一点。协议(protocol)的专门版本(即具有已知关联类型的协议(protocol)引用)仍然不是具体类型。我不会为这种能力屏住呼吸。我认为这是我提出的解决方案中最弱的一个。
rawValue
上提供,我想为什么不实现我自己的通用镜像:struct RawRepresentableMirror<T: RawRepresentable>: MirrorType {
private let realValue: T
init(_ value: T) {
realValue = value
}
var value: Any { return realValue }
var valueType: Any.Type { return T.self }
var objectIdentifier: ObjectIdentifier? { return nil }
var disposition: MirrorDisposition { return .Enum }
var count: Int { return 1 }
subscript(index: Int) -> (String, MirrorType) {
switch index {
case 0:
return ("rawValue", reflect(realValue.rawValue))
default:
fatalError("Index out of range")
}
}
var summary: String {
return "Raw Representable Enum: \(realValue)"
}
var quickLookObject: QuickLookObject? {
return QuickLookObject.Text(summary)
}
}
RawRepresentable
成为
Reflectable
这样我们就可以先投
theEnum as Reflectable
(不需要关联类型)然后拨打
reflect(theEnum)
给我们我们很棒的定制镜子:
extension RawRepresentable: Reflectable {
func getMirror() -> MirrorType {
return RawRepresentableMirror(self)
}
}
Compiler error: Extension of protocol 'RawRepresentable' cannot have an inheritance clause
extension MyClass: MyProtocol {
// Conform to new protocol
}
extension MyProtocol {
// Default implementations for MyProtocol
}
protocol AnyRawRepresentable {
var anyRawValue: Any { get }
}
extension RawRepresentable: AnyRawRepresentable {
var anyRawValue: Any {
return rawValue
}
}
RawRepresentable
的类型,请使用此默认实现使该类型也符合
AnyRawRepresentable
。”
AnyRawRepresentable
不会有关联的类型要求,但仍然可以检索
rawValue
作为
Any
.在我们的代码中,然后:
if let anyRawEnum = theEnum as? AnyRawRepresentable { // Able to cast to
let anyRawValue = anyRawEnum.anyRawValue // anyRawValue is of type Any
switch anyRawValue {
case let s as String:
return s
case let csc as CustomStringConvertible:
return csc.description
default:
return nil
}
}
Mirror
的回复。在
RawRepresentable
枚举不包含其
rawValue
.至于选项 #2 和 #3,它们仍然在 Swift future 版本的可能性范围内。它们已在 Swift 邮件列表和
Generics Manifesto 中提及。由 Swift 团队的 Doug Gregor 撰写的文档。
Any
其中有关联类型或允许使用如下语法:
anyEnum as? Any<RawRepresentable where .RawValue == String>
关于swift - 从泛型函数中的枚举获取 rawValue,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/31607629/
我是一名优秀的程序员,十分优秀!