protocol
IteratorProtocol
A type that supplies the values of a sequence one at a time.
Instance Methods
Advances to the next element and returns it, or nil
if no next element
exists.
Repeatedly calling this method returns, in order, all the elements of the
underlying sequence. As soon as the sequence has run out of elements, all
subsequent calls return nil
.
You must not call this method if any other copy of this iterator has been
advanced with a call to its next()
method.
The following example shows how an iterator can be used explicitly to
emulate a for
-in
loop. First, retrieve a sequence's iterator, and
then call the iterator's next()
method until it returns nil
.
let
numbers
= [
2
,
3
,
5
,
7
]
var
numbersIterator
=
numbers
.
makeIterator
()
while
let
num
=
numbersIterator
.
next
() {
(
num
)
}
// Prints "2"
// Prints "3"
// Prints "5"
// Prints "7"
Declaration
mutating
func
next
() -
>
Self
.
Element
?
The
IteratorProtocol
protocol is tightly linked with theSequence
protocol. Sequences provide access to their elements by creating an iterator, which keeps track of its iteration process and returns one element at a time as it advances through the sequence.Whenever you use a
for
-in
loop with an array, set, or any other collection or sequence, you're using that type's iterator. Swift uses a sequence's or collection's iterator internally to enable thefor
-in
loop language construct.Using a sequence's iterator directly gives you access to the same elements in the same order as iterating over that sequence using a
for
-in
loop. For example, you might typically use afor
-in
loop to print each of the elements in an array.Behind the scenes, Swift uses the
animals
array's iterator to loop over the contents of the array.The call to
animals.makeIterator()
returns an instance of the array's iterator. Next, thewhile
loop calls the iterator'snext()
method repeatedly, binding each element that is returned toanimal
and exiting when thenext()
method returnsnil
.Using Iterators Directly
You rarely need to use iterators directly, because a
for
-in
loop is the more idiomatic approach to traversing a sequence in Swift. Some algorithms, however, may call for direct iterator use.One example is the
reduce1(_:)
method. Similar to thereduce(_:_:)
method defined in the standard library, which takes an initial value and a combining closure,reduce1(_:)
uses the first element of the sequence as the initial value.Here's an implementation of the
reduce1(_:)
method. The sequence's iterator is used directly to retrieve the initial value before looping over the rest of the sequence.The
reduce1(_:)
method makes certain kinds of sequence operations simpler. Here's how to find the longest string in a sequence, using theanimals
array introduced earlier as an example:Using Multiple Iterators
Whenever you use multiple iterators (or
for
-in
loops) over a single sequence, be sure you know that the specific sequence supports repeated iteration, either because you know its concrete type or because the sequence is also constrained to theCollection
protocol.Obtain each separate iterator from separate calls to the sequence's
makeIterator()
method rather than by copying. Copying an iterator is safe, but advancing one copy of an iterator by calling itsnext()
method may invalidate other copies of that iterator.for
-in
loops are safe in this regard.Adding IteratorProtocol Conformance to Your Type
Implementing an iterator that conforms to
IteratorProtocol
is simple. Declare anext()
method that advances one step in the related sequence and returns the current element. When the sequence has been exhausted, thenext()
method returnsnil
.For example, consider a custom
Countdown
sequence. You can initialize theCountdown
sequence with a starting integer and then iterate over the count down to zero. TheCountdown
structure's definition is short: It contains only the starting count and themakeIterator()
method required by theSequence
protocol.The
makeIterator()
method returns another custom type, an iterator namedCountdownIterator
. TheCountdownIterator
type keeps track of both theCountdown
sequence that it's iterating and the number of times it has returned a value.Each time the
next()
method is called on aCountdownIterator
instance, it calculates the new next value, checks to see whether it has reached zero, and then returns either the number, ornil
if the iterator is finished returning elements of the sequence.Creating and iterating over a
Countdown
sequence uses aCountdownIterator
to handle the iteration.