How to access a Swift enum associated value outside of a switch statement

后端 未结 5 1670
我寻月下人不归
我寻月下人不归 2021-01-31 07:08

Consider:

enum Line {
    case    Horizontal(CGFloat)
    case    Vertical(CGFloat)
}

let leftEdge             =  Line.Horizontal(0.0)
let leftMaskRightEdge             


        
相关标签:
5条回答
  • 2021-01-31 07:46

    I think you may be trying to use enum for something it was not intended for. The way to access the associated values is indeed through switch as you've done, the idea being that the switch always handles each possible member case of the enum.

    Different members of the enum can have different associated values (e.g., you could have Diagonal(CGFloat, CGFloat) and Text(String) in your enum Line), so you must always confirm which case you're dealing with before you can access the associated value. For instance, consider:

    enum Line {
        case Horizontal(CGFloat)
        case Vertical(CGFloat)
        case Diagonal(CGFloat, CGFloat)
        case Text(String)
    }
    var myLine = someFunctionReturningEnumLine()
    let value = myLine.associatedValue // <- type?
    

    How could you presume to get the associated value from myLine when you might be dealing with CGFloat, String, or two CGFloats? This is why you need the switch to first discover which case you have.

    In your particular case it sounds like you might be better off with a class or struct for Line, which might then store the CGFloat and also have an enum property for Vertical and Horizontal. Or you could model Vertical and Horizontal as separate classes, with Line being a protocol (for example).

    0 讨论(0)
  • 2021-01-31 07:47

    With Swift 2 it's possible to get the associated value (read only) using reflection.

    To make that easier just add the code below to your project and extend your enum with the EVAssociated protocol.

        public protocol EVAssociated {
        }
    
        public extension EVAssociated {
            public var associated: (label:String, value: Any?) {
                get {
                    let mirror = Mirror(reflecting: self)
                    if let associated = mirror.children.first {
                        return (associated.label!, associated.value)
                    }
                    print("WARNING: Enum option of \(self) does not have an associated value")
                    return ("\(self)", nil)
                }
            }
        }
    

    Then you can access the .asociated value with code like this:

        class EVReflectionTests: XCTestCase {
                func testEnumAssociatedValues() {
                    let parameters:[EVAssociated] = [usersParameters.number(19),
    usersParameters.authors_only(false)]
                let y = WordPressRequestConvertible.MeLikes("XX", Dictionary(associated: parameters))
                // Now just extract the label and associated values from this enum
                let label = y.associated.label
                let (token, param) = y.associated.value as! (String, [String:Any]?)
    
                XCTAssertEqual("MeLikes", label, "The label of the enum should be MeLikes")
                XCTAssertEqual("XX", token, "The token associated value of the enum should be XX")
                XCTAssertEqual(19, param?["number"] as? Int, "The number param associated value of the enum should be 19")
                XCTAssertEqual(false, param?["authors_only"] as? Bool, "The authors_only param associated value of the enum should be false")
    
                print("\(label) = {token = \(token), params = \(param)")
            }
        }
    
        // See http://github.com/evermeer/EVWordPressAPI for a full functional usage of associated values
        enum WordPressRequestConvertible: EVAssociated {
            case Users(String, Dictionary<String, Any>?)
            case Suggest(String, Dictionary<String, Any>?)
            case Me(String, Dictionary<String, Any>?)
            case MeLikes(String, Dictionary<String, Any>?)
            case Shortcodes(String, Dictionary<String, Any>?)
        }
    
        public enum usersParameters: EVAssociated {
            case context(String)
            case http_envelope(Bool)
            case pretty(Bool)
            case meta(String)
            case fields(String)
            case callback(String)
            case number(Int)
            case offset(Int)
            case order(String)
            case order_by(String)
            case authors_only(Bool)
            case type(String)
        }
    

    The code above is from my project https://github.com/evermeer/EVReflection https://github.com/evermeer/EVReflection

    0 讨论(0)
  • 2021-01-31 07:49

    Why this is not possible is already answered, so this is only an advice. Why don't you implement it like this. I mean enums and structs are both value types.

    enum Orientation {
        case Horizontal
        case Vertical
    }
    
    struct Line {
    
        let orientation : Orientation
        let value : CGFloat
    
        init(_ orientation: Orientation, _ value: CGFloat) {
    
            self.orientation = orientation
            self.value = value
        }
    } 
    
    let x = Line(.Horizontal, 20.0)
    
    // if you want that syntax 'Line.Horizontal(0.0)' you could fake it like this
    
    struct Line {
    
        let orientation : Orientation
        let value : CGFloat
    
        private init(_ orientation: Orientation, _ value: CGFloat) {
    
            self.orientation = orientation
            self.value = value
        }
    
        static func Horizontal(value: CGFloat) -> Line { return Line(.Horizontal, value) }
        static func Vertical(value: CGFloat) -> Line { return Line(.Vertical, value) }
    }
    
    let y = Line.Horizontal(20.0)
    
    0 讨论(0)
  • 2021-01-31 07:51

    As others have pointed out, this is now kind of possible in Swift 2:

    import CoreGraphics
    
    enum Line {
        case    Horizontal(CGFloat)
        case    Vertical(CGFloat)
    }
    
    let min = Line.Horizontal(0.0)
    let mid = Line.Horizontal(0.5)
    let max = Line.Horizontal(1.0)
    
    func doToLine(line: Line) -> CGFloat? {
        if case .Horizontal(let value) = line {
            return value
        }
        return .None
    }
    
    doToLine(min) // prints 0
    doToLine(mid) // prints 0.5
    doToLine(max) // prints 1
    
    0 讨论(0)
  • 2021-01-31 07:56

    You can use a guard statement to access the associated value, like this.

    enum Line {
        case    Horizontal(Float)
        case    Vertical(Float)
    }
    
    let leftEdge             =  Line.Horizontal(0.0)
    let leftMaskRightEdge    =  Line.Horizontal(0.05)
    
    guard case .Horizontal(let leftEdgeValue) = leftEdge else { fatalError() }
    
    print(leftEdgeValue)
    
    0 讨论(0)
提交回复
热议问题