Swift is known for its simplicity and expressive syntax, but some incredibly powerful functions often go unnoticed. These lesser-known built-in functions can help solve common programming challenges more efficiently, making your code more readable, concise, and performant.

let’s explore a set of Swift functions that will help you write cleaner, more elegant solutions to everyday problems.

What we’ll cover

  • Smarter iteration and loops with stride
  • combining sequence and parallel Iteration with zip
  • Quick validation with allSatisfy(_:)
  • Lazy Sequence Generation with sequence
  • Categorizing data with Dictionary(grouping:by:)

Smarter iteration and loops with stride

Stride function provides a flexible way to create sequences of values based on a specific range and step, making it useful for various looping and iteration scenarios in Swift.

Swift provides two variations of stride:

  • stride(from:to:by:) Excludes the end value (to).
  • stride(from:through:by:) Includes the end value (through).

1. Creating a Range of Values with a Custom Step

// Excluding the last value
for value in stride(from: 0, to: 10, by: 2) {
    print(value)  // 0, 2, 4, 6, 8
}

// OR
let eventNumber: [Int] = stride(from: 0, to: 10, by: 2).map(\.self)
print(eventNumber) // [0, 2, 4, 6, 8]

// Including the last value
let eventNumberIncusive: [Int] = stride(from: 0, through: 10, by: 2).map(\.self) // inluding last element ( through )
print(eventNumberIncusive) // [0, 2, 4, 6, 8, 10]

2. Generating Reverse Sequences, iterate backward

let countdown = stride(from: 10, to: 0, by: -3).map { $0 }
print(countdown) // [10, 7, 4, 1]

3. Works with floating-point numbers

for value in stride(from: 0.0, through: 1.0, by: 0.2) {
    print(value) // 0.0, 0.2, 0.4, 0.6, 0.8, 1.0
}

4. Can be combined with higher-order functions like map, filter, and reduce for more data transformations

let squaredNumbers = stride(from: 1, through: 10, by: 2).map { $0 * $0 }
print(squaredNumbers) // [1, 9, 25, 49, 81]

zip(_:_:) – for combining sequence and parallel Iteration:

The zip function can be used to combine two or more sequences into a single sequence of tuples or iterating over two related arrays simultaneously

1. Combining Two Sequences into a Single Sequence of Tuples

let names = ["Alice", "Bob", "Charlie"]
let scores = [85, 92, 78]

let zipped = zip(names, scores)
zipped.forEach { print($0) } // ("Alice", 85), ("Bob", 92), ("Charlie", 78)

2. Iterate two collections in sync without manual indexing.

let letters = ["a", "b", "c"]
let numbers = [1, 2, 3]

for (letter, number) in zip(letters, numbers) {
    print("\(letter) - \(number)") // a - 1, b - 2, c - 3
}

Quick validation with allSatisfy(_:)

allSatisfy(_:) check whether all elements in a collection meet a specific condition, making validations more concise.

let numbers = [2, 4, 6, 8]
let allEven = numbers.allSatisfy { $0 % 2 == 0 }  // true

Lazy Sequence Generation with sequence

sequence is usefull for generating sequences lazily through the sequence function family. These functions are invaluable for creating memory-efficient, on-demand sequences.

Swift provides two variations of sequence:

  • sequence(first:next:) creates a sequence starting from an initial value and repeatedly applies a function to generate the next value until it returns nil.
  • sequence(state:next:) function maintains a state variable that you can mutate, making it useful for generating sequences with complex state tracking.

1. Finite squence with sequence(first:next:), stops when nil is returned

let countdown = sequence(first: 5, next: { $0 > 0 ? $0 - 1 : nil })
// 5, 4, 3, 2, 1, 0

2. Infinite squence with sequence(first:next:)

llet powersOfTwo = sequence(first: 1, next: { $0 * 2 })
print(Array( powersOfTwo.prefix(5))) // [1, 2, 4, 8, 16]

3. Finite squence with sequence(state:next:), stops when nil is returned

let countingUpToFive = sequence(state: 1) { state -> Int? in
    if state > 5 { return nil } // exit condition
    let result = state
    state = state + 1 // updated ( next ) state
    return result // previous state
}

print(Array(countingUpToFive)) // [1, 2, 3, 4, 5]

4. Infinite fibonacci squence with sequence(state:next:)

let fibonacci = sequence(state: (0, 1)) { state -> Int? in
    let result = state.0 // previous stat, we want the sequence to start with first element of initial/first state ( 0, 1)
    state = (state.1, state.0 + state.1) // updated ( next ) state
    return result // previous state
}

print(Array(fibonacci.prefix(10))) // [0, 1, 1, 2, 3, 5, 8, 13, 21, 34]

Categorizing data with Dictionary(grouping:by:)

Dictionary(grouping:by:)lets you group elements from a collection ( array or sequence ) into a dictionary based on a common key, making it great for categorizing data without manually iterating and sorting elements.

let words = ["apple", "banana", "avocado", "blueberry"]
let grouped = Dictionary(grouping: words, by: { $0.first! })
print(grouped) // ["a": ["apple", "avocado"], "b": ["banana", "blueberry"]]
let numbers = [1, 2, 3, 4, 5, 6, 7, 8, 9, 10]
let groupedNumbers = Dictionary(grouping: numbers, by: { $0 % 2 == 0 ? "Even" : "Odd" })
print(groupedNumbers) // ["Even": [2, 4, 6, 8, 10], "Odd": [1, 3, 5, 7, 9]]

Conclusion

Competitive programming and real-world app development demand efficiency and clarity. Swift isn’t just about syntax simplicity—it’s packed with hidden features that can significantly improve your code’s efficiency and readability. By integrating these powerful built-in functions into your workflow, you can eliminate unnecessary loops, reduce complexity, and enhance performance. Next time you tackle a complex problem, think about using stride for iteration, zip for parallel sequences, sequence for lazy evaluation, and Dictionary(grouping:by:) for effortless categorization.

If you think i missed anything, let me know and feel free to share with me any thoughts or feedback you have.

Thanks for reading!🚀