Enum'a ham bir Int değeri verirseniz, döngü çok daha kolay olacaktır.
Örneğin, anyGenerator
değerleriniz genelinde numaralandırılabilen bir jeneratör elde etmek için kullanabilirsiniz :
enum Suit: Int, CustomStringConvertible {
case Spades, Hearts, Diamonds, Clubs
var description: String {
switch self {
case .Spades: return "Spades"
case .Hearts: return "Hearts"
case .Diamonds: return "Diamonds"
case .Clubs: return "Clubs"
}
}
static func enumerate() -> AnyGenerator<Suit> {
var nextIndex = Spades.rawValue
return anyGenerator { Suit(rawValue: nextIndex++) }
}
}
// You can now use it like this:
for suit in Suit.enumerate() {
suit.description
}
// or like this:
let allSuits: [Suit] = Array(Suit.enumerate())
Ancak, bu oldukça yaygın bir örüntüye benziyor, sadece bir protokole uyarak herhangi bir enum tipini numaralandırırsak hoş olmaz mıydı? Swift 2.0 ve protokol uzantıları ile şimdi yapabiliriz!
Bunu projenize eklemeniz yeterlidir:
protocol EnumerableEnum {
init?(rawValue: Int)
static func firstValue() -> Int
}
extension EnumerableEnum {
static func enumerate() -> AnyGenerator<Self> {
var nextIndex = firstRawValue()
return anyGenerator { Self(rawValue: nextIndex++) }
}
static func firstRawValue() -> Int { return 0 }
}
Artık bir enum oluşturduğunuzda (Int ham değeri olduğu sürece), protokole uyarak numaralandırılabilir hale getirebilirsiniz:
enum Rank: Int, EnumerableEnum {
case Ace, Two, Three, Four, Five, Six, Seven, Eight, Nine, Ten, Jack, Queen, King
}
// ...
for rank in Rank.enumerate() { ... }
Numaralandırma değerleriniz 0
(varsayılan) ile başlamazsa , firstRawValue
yöntemi geçersiz kılın :
enum DeckColor: Int, EnumerableEnum {
case Red = 10, Blue, Black
static func firstRawValue() -> Int { return Red.rawValue }
}
// ...
let colors = Array(DeckColor.enumerate())
Daha standart CustomStringConvertible protokolüsimpleDescription
ile değiştirmeyi de içeren son Suit sınıfı aşağıdaki gibi görünecektir:
enum Suit: Int, CustomStringConvertible, EnumerableEnum {
case Spades, Hearts, Diamonds, Clubs
var description: String {
switch self {
case .Spades: return "Spades"
case .Hearts: return "Hearts"
case .Diamonds: return "Diamonds"
case .Clubs: return "Clubs"
}
}
}
// ...
for suit in Suit.enumerate() {
print(suit.description)
}
Swift 3 sözdizimi:
protocol EnumerableEnum {
init?(rawValue: Int)
static func firstRawValue() -> Int
}
extension EnumerableEnum {
static func enumerate() -> AnyIterator<Self> {
var nextIndex = firstRawValue()
let iterator: AnyIterator<Self> = AnyIterator {
defer { nextIndex = nextIndex + 1 }
return Self(rawValue: nextIndex)
}
return iterator
}
static func firstRawValue() -> Int {
return 0
}
}