3

我希望实现某种风格的枚举语法/功能,但是我不确定如何实现它。目前我有以下内容:

internal enum Timeframe: Equatable {

    // MARK: - Cases

    case hour(count: Int)
    case day(count: Int)
    case week(count: Int)
    case month(count: Int)
    case year(count: Int)
    case all
    case exact(date: Date)
    case unspecified
}

我想消除与计数相关的值,除非它是必需的。例如:

let oneDayTimeframe: Timeframe = .day
let twoDayTimeframe: Timeframe = .day.multiplied(by: 2)

这可能吗?即使没有办法完全实现我正在寻找的东西,我也会感谢有关潜在改进的建议。在大多数情况下,我最终使用 (count: 1),这似乎有点冗长。如果默认值可与关联值一起使用,我会使用

case hour(count: Int = 1)

有什么建议么?

4

2 回答 2

3

对您提供的答案进行一些迭代:

enum TimeFrame: Equatable {

    case all
    case countable(timeFrame: CountableTimeFrame)
    case exact(date: Date)
}

enum CountableTimeFrame: Equatable {

    case hour
    case day
    case week
    case month
    case year

    indirect case multiple(CountableTimeFrame, Int)

    var timeFrame: TimeFrame {
        return .countable(timeFrame: self)
    }

    static func * (left: CountableTimeFrame, right: Int) -> CountableTimeFrame {
        switch left {
        case .multiple(let timeFrame, let count):
            return .multiple(timeFrame, count * right)
        default:
            return .multiple(left, right)
        }
    }

    static func * (left: Int, right: CountableTimeFrame) -> CountableTimeFrame {
        return right * left
    }
}

将禁止滥用,例如禁止:

let timeFrame: TimeFrame = .multiple(.exact(date: someDate), 666)
let timeFrame: TimeFrame = .multiple(unspecified, 40)
let timeFrame: TimeFrame = .multiple(all, -1) 

并允许使用 * 运算符进行乘法运算,例如

let timeFrame: CountableTimeFrame = 4 * .hour
print(timeFrame) // multiple(__lldb_expr_5.CountableTimeFrame.hour, 4)
print(timeFrame * 2) // multiple(__lldb_expr_5.CountableTimeFrame.hour, 8)

和 .unspecified:

let optionalTimeFrame: TimeFrame? = nil
于 2019-03-02T13:03:52.633 回答
0

我发现使用递归枚举可以做到这一点。尽管与我建议的不完全相同,但这实现了类似的结果:

internal enum Timeframe: Equatable {

    // MARK: - Cases

    case hour
    case day
    case week
    case month
    case year
    case all
    case exact(date: Date)
    case unspecified

    // MARK: - Indirect Cases

    indirect case multiple(Timeframe, Int)
}
于 2019-03-01T22:26:55.503 回答