Подтвердить что ты не робот

Как разделить массив пополам в Swift?

Как мне разделить колоду карт? У меня есть массив и случайный карточный дилер, но я понятия не имею, как разделить колоду.

Спасибо всем за помощь! У меня теперь есть приложение для рабочих карт, они столкнулись с другими проблемами, но они были быстро решены.

4b9b3361

Ответ 1

Вы можете сделать расширение, чтобы он мог возвращать массив из двух массивов, работающих с Ints, Strings и т.д.:

extension Array {
    func split() -> [[Element]] {
        let ct = self.count
        let half = ct / 2
        let leftSplit = self[0 ..< half]
        let rightSplit = self[half ..< ct]
        return [Array(leftSplit), Array(rightSplit)]
    }
}

let deck = ["J", "Q", "K", "A"]
let nums = [0, 1, 2, 3, 4]

deck.split() // [["J", "Q"], ["K", "A"]]
nums.split() // [[0, 1], [2, 3, 4]]

Но возврат именованного кортежа еще лучше, потому что он обеспечивает тот факт, что в результате вы ожидаете ровно два массива:

extension Array {
    func split() -> (left: [Element], right: [Element]) {
        let ct = self.count
        let half = ct / 2
        let leftSplit = self[0 ..< half]
        let rightSplit = self[half ..< ct]
        return (left: Array(leftSplit), right: Array(rightSplit))
    }
}

let deck = ["J", "Q", "K", "A"]

let splitDeck = deck.split()
print(splitDeck.left) // ["J", "Q"]
print(splitDeck.right) // ["K", "A"]

Примечание: кредиты передаются Андрею и Qbyte за первый правильный ответ, я просто добавляю информацию.

Ответ 2

Вы можете использовать диапазон индексов

let deck: [String] = ["J", "Q", "K", "A"]

// use ArraySlice only for transient computation
let leftSplit: ArraySlice<String> = deck[0 ..< deck.count / 2] // "J", "Q"
let rightSplit: ArraySlice<String> = deck[deck.count / 2 ..< deck.count] // "K", "A"

// make arrays from ArraySlice
let leftDeck: [String] = Array(leftSplit) // "J", "Q"
let rightDeck: [String] = Array(rightSplit) // "K", "A"

EDIT: код выше для Swift 2, возможно, для Swift 3 это более удобный способ.

Ответ 3

Swift

Более общее решение для разделения массива на куски - ответ по этой ссылке

extension Array {
    func chunked(into size: Int) -> [[Element]] {
        return stride(from: 0, to: count, by: size).map {
            Array(self[$0 ..< Swift.min($0 + size, count)])
        }
    }
}
let numbers = Array(1...100)
let result = numbers.chunked(into: 5)

Ответ 4

Вы можете создать расширение на SequenceType и создать функцию с именем divide.

Эта функция будет перебирать элементы последовательности, помещая те, которые соответствуют предикату, в один массив (slice) и те, которые не совпадают с другим массивом (remainder).

Функция возвращает кортеж, содержащий slice и remainder.

extension SequenceType {

    /**
     Returns a tuple with 2 arrays.
     The first array (the slice) contains the elements of self that match the predicate.
     The second array (the remainder) contains the elements of self that do not match the predicate.
     */
    func divide(@noescape predicate: (Self.Generator.Element) -> Bool) -> (slice: [Self.Generator.Element], remainder: [Self.Generator.Element]) {
        var slice:     [Self.Generator.Element] = []
        var remainder: [Self.Generator.Element] = []
        forEach {
            switch predicate($0) {
            case true  : slice.append($0)
            case false : remainder.append($0)
            }
        }
        return (slice, remainder)
    }

}

Это пример

let tuple = [1, 2, 3, 4, 5].divide({ $0 >= 3 })
tuple.slice     // [3, 4, 5]
tuple.remainder // [1, 2]

Ответ 5

И еще одна реализация ранее представленных идей. Во-первых, до текущей документации Swift лучше выбирать имена в прошедшем простом времени для функций, которые дают некоторый результат, и настоящее время для мутирующих. Во-вторых, для меня лучше выбрать половину добавления count% 2, чтобы получить более равномерный результат.

Вот оно:

extension Array {
    func splitted() -> ([Element], [Element]) {
        let half = count / 2 + count % 2
        let head = self[0..<half]
        let tail = self[half..<count]

        return (Array(head), Array(tail))
    }
}

И результаты:

let set1 = [1, 2, 3, 4, 5, 6, 7,8]
let set2 = [1, 2, 3, 4, 5]
let set3 = [1]
let set4 = [Int]()

print(set1.splitted())
print(set2.splitted())
print(set3.splitted())
print(set4.splitted())

([1, 2, 3, 4], [5, 6, 7, 8])
([1, 2, 3], [4, 5])
([1], [])
([], [])