当前位置: 首页 > 文档资料 > Swift 编程语言 >

内嵌类型

优质
小牛编辑
132浏览
2023-12-01

枚举通常用于实现特定类或结构体的功能。类似的,它也可以在更加复杂的类型环境中方便的定义通用类和结构体。为实现这种功能,Swift 允许你定义内嵌类型,借此在支持类型的定义中嵌套枚举、类、或结构体。

若要在一种类型中嵌套另一种类型,在其支持类型的大括号内定义即可。可以根据需求多级嵌套数个类型。

内嵌类型的使用

下方的例子定义了一个名为 BlackJackCard 的结构体,模拟了21点游戏中的扑克牌。 BlackjackCard 结构体包含两个内嵌的枚举类型 Suit 和 Rank 。

在21点游戏中,Ace 可以表示一或十一两个值,这通过 Rank 枚举中内嵌的结构体 Values 决定:

struct BlackjackCard {

    // nested Suit enumeration
    enum Suit: Character {
        case spades = "♠", hearts = "♡", diamonds = "♢", clubs = "♣"
    }

    // nested Rank enumeration
    enum Rank: Int {
        case two = 2, three, four, five, six, seven, eight, nine, ten
        case jack, queen, king, ace
        struct Values {
            let first: Int, second: Int?
        }
        var values: Values {
            switch self {
            case .ace:
                return Values(first: 1, second: 11)
            case .jack, .queen, .king:
                return Values(first: 10, second: nil)
            default:
                return Values(first: self.rawValue, second: nil)
            }
        }
    }

    // BlackjackCard properties and methods
    let rank: Rank, suit: Suit
    var description: String {
        var output = "suit is \(suit.rawValue),"
        output += " value is \(rank.values.first)"
        if let second = rank.values.second {
            output += " or \(second)"
        }
        return output
    }
}

 

Suit 枚举用于描述扑克牌的四种花色,并用原始值 Character 来代表各自的花色。

Rank 枚举用于描述扑克牌可能出现的十三种点数,并用原始值 Int 来代表各自的点数值(这里的 Int 并不会用于 J、Q、K、Ace 的表示)。

如上所述, Rank 枚举中定义了一个内嵌结构体 Values 。这个结构体封装了大多牌只有一个值,而 Ace 可以有两个值这一事实。 Values 结构体定义了两个属性来表示这些:

  • Int 类型的 first
  • Int? 类型的 second ,或者说“可选 Int ”

Rank 还定义了一个计算属性, values ,它用于返回 Values 结构体的实例。这个计算属性会根据牌的点数,用适当的值初始化新的 Values 实例。对于 Jack 、 Queen 、 King 、和 Ace 使用特殊的值。而对于数值的牌,则使用它本身的 Int 原始值。

BlackjackCard 结构体本身有两个属性—— rank 和 suit 。还定义了一个名为 description 的计算属性,用 rank 和 suit 储存的值构建对扑克牌花色和值的描述。 description 属性使用可选绑定来检查是否有第二个值要描述,若有,则添加对第二个值的描述。

由于 BlackjackCard 是一个没有自定义初始化器的结构体,如结构体类型的成员初始化器所述,它有一个隐式的成员初始化器。你可以使用这个初始化器去初始化新的常量 theAceOfSpades :

let theAceOfSpades = BlackjackCard(rank: .ace, suit: .spades)
print("theAceOfSpades: \(theAceOfSpades.description)")
// Prints "theAceOfSpades: suit is ♠, value is 1 or 11"

尽管 Rank 和 Suit 被嵌套在 BlackjackCard 中,但其类型仍可从上下文中推断出来,因此,该实例的初始化器可以单独通过成员名称( .ace 和 .spades )引用枚举类型。在上面的例子中, description 属性正确的反馈了黑桃 Ace 拥有 1 或 11 两个值。

引用内嵌类型

要在定义外部使用内嵌类型,只需在其前缀加上内嵌了它的类的类型名即可:

let heartsSymbol = BlackjackCard.Suit.hearts.rawValue
// heartsSymbol is "♡"

对于上面的栗子来说,可以使 Suit 、 Rank 和 Values 的名字尽可能的短,因为它们的名字由定义时的上下文自然限定。