gpt4 book ai didi

arrays - 如何为特定类型的数组扩展数组功能?

转载 作者:可可西里 更新时间:2023-11-01 00:52:09 26 4
gpt4 key购买 nike

(以我常用的扑克牌为例)
我正在尝试制作一个通用的 CardCollection这两个都是Deck和一个 Hand会继承自。 Decks 和 Hands 都需要排序或洗牌,但会有一些差异,例如初始化以及删除 Card 的方法是否正确。在别处使用的是 Deal (对于 Deck ),Play , 或 Discard (对于 Hand 秒)。

class CardCollection: <Some protocol or another that Arrays use> {
var collective = [Card]()
// CardCollection-specific functions

// pass-through functions
func append(newCard: Card) {
collective.append(newCard)
}
}


class Deck: CardCollection {
// deck-specific functions
}
class Hand: CardCollection {
// hand-specific functions
}

我目前实现它的方式(见上文)是使用一个包含卡片数组的类,但如果不编写大量传递函数来获取我的类,我就不能像使用数组一样使用我的类以数组的形式遵守所有协议(protocol)。

我需要的是一种能让我做类似 for card in deck 的方法(好像 deck 只是一个 Array<Card> )而无需编写大量的包装函数来获得 CardCollection遵守所有必要的协议(protocol)。

我如何制作一个像 Array<Card> 一样运行的 CardCollection?无需对 Array 协议(protocol)使用的每个函数执行传递函数用途?

最佳答案

你可以定义一个 CardCollection 协议(protocol)继承自RangeReplaceableCollectionType,和一个带有默认实现的协议(protocol)扩展来转发所有访问底层 collective 数组的方法:

struct Card { 
// Simplified for demonstration purposes:
let rank : Int
let suit : Int
}

protocol CardCollection : RangeReplaceableCollectionType {
var collective : [Card] { get set }
}

extension CardCollection {

var startIndex : Int { return collective.startIndex }
var endIndex : Int { return collective.endIndex }

subscript(position : Int) -> Card {
get {
return collective[position]

}
set(newElement) {
collective[position] = newElement
}
}

mutating func replaceRange<C : CollectionType where C.Generator.Element == Card>(subRange: Range<Int>, with newElements: C) {
collective.replaceRange(subRange, with: newElements)
}
}

然后

struct Deck: CardCollection {
var collective = [Card]()

}

struct Hand: CardCollection {
var collective = [Card]()

}

都符合 RangeReplaceableCollectionType 并且可以被处理像一个数组:

var deck = Deck()
deck.append(Card(rank: 1, suit: 1))
deck[0] = Card(rank: 2, suit: 3)

for card in deck {
print(card)
}

var hand = Hand()
hand.append(deck.first!)

如果 Deck/Hand而不是结构,那么它们需要是 final 或有一个 required init() 方法,相比 Why use required Initializers in Swift classes? .


稍微通用一点,可以定义一个ElementCollection协议(protocol)(独立于 Card 类型)它的行为就像一个数组(通过符合RangeReplaceableCollectionType) 并将访问转发到底层 elements 数组:

protocol ElementCollection : RangeReplaceableCollectionType {
typealias Element
var elements : [Element] { get set }
}

extension ElementCollection {

var startIndex : Int { return elements.startIndex }
var endIndex : Int { return elements.endIndex }

subscript(position : Int) -> Element {
get {
return elements[position]

}
set(newElement) {
elements[position] = newElement
}
}

mutating func replaceRange<C : CollectionType where C.Generator.Element == Element>(subRange: Range<Int>, with newElements: C) {
elements.replaceRange(subRange, with: newElements)
}
}

然后用作

struct Card { 
// Simplified for demonstration purposes:
let rank : Int
let suit : Int
}

struct Deck: ElementCollection {
var elements = [Card]()

}

struct Hand: ElementCollection {
var elements = [Card]()

}

关于arrays - 如何为特定类型的数组扩展数组功能?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/34237047/

26 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com