Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

Efficient way of downsample collection via decimating or extracting every nth element in Swift

I'm trying to downsample a long collection by decimating or extracting every nth element.

Here's what I got for my array extension:

func downsampled(to threshold: Int) -> [T] {
    // Validate that threshold falls in valid range
    guard !isEmpty, 1...count ~= threshold else { return Array(self) }
    
    let skip = (count / threshold) + 1
    var index = 0
    
    var items = [T]()
    while index < count {
        items.append(self[index])
        index += skip
    }
    
    return items
}

I'm expecting 50-100k items in the original array and will probably downsample to the native bounds width of the screen (500-1k points).

Is there a more concise or efficient way of doing this?

like image 316
TruMan1 Avatar asked Dec 06 '25 19:12

TruMan1


1 Answers

extension RangeReplaceableCollection {
    func every(from: Index? = nil, through: Index? = nil, nth: Int) -> Self { .init(stride(from: from, through: through, by: nth)) }
}

extension Collection {
    func stride(from: Index? = nil, through: Index? = nil, by: Int) -> AnySequence<Element> {
        var index = from ?? startIndex
        let endIndex = through ?? self.endIndex
        return AnySequence(AnyIterator {
            guard index < endIndex else { return nil }
            defer { index = self.index(index, offsetBy: by, limitedBy: endIndex) ?? endIndex }
            return self[index]
        })
    }
}

Playground testing

let array = [1,2,3,4,5,6,7,8,9,10,11,12,13,14,15]
for element in array.stride(by: 3) {
    print(element)
}
array.stride(by: 3).forEach {
    print($0)
}
let nth = array.every(nth: 3)  // [1, 4, 7, 10, 13]

let str = "0123456789"
for character in str.stride(by: 2) {
    print(character)
}
str.stride(by: 2).forEach {
    print($0)
}
let even = str.every(nth: 2)   // "02468"
like image 96
Leo Dabus Avatar answered Dec 09 '25 15:12

Leo Dabus