Swift - How to mutate a struct object when iterating over it
Solution 1
struct
are value types, thus in the for
loop you are dealing with a copy.
Just as a test you might try this:
Swift 3:
struct Options {
var backgroundColor = UIColor.black
}
var arrayOfMyStruct = [Options]()
for (index, _) in arrayOfMyStruct.enumerated() {
arrayOfMyStruct[index].backgroundColor = UIColor.red
}
Swift 2:
struct Options {
var backgroundColor = UIColor.blackColor()
}
var arrayOfMyStruct = [Options]()
for (index, _) in enumerate(arrayOfMyStruct) {
arrayOfMyStruct[index].backgroundColor = UIColor.redColor()
}
Here you just enumerate the index, and access directly the value stored in the array.
Hope this helps.
Solution 2
You can use use Array.indices
:
for index in arrayOfMyStruct.indices {
arrayOfMyStruct[index].backgroundColor = UIColor.red
}
Solution 3
You are working with struct
objects which are copied to local variable when using for in
loop. Also array is a struct
object, so if you want to mutate all members of the array, you have to create modified copy of original array filled by modified copies of original objects.
arrayOfMyStruct = arrayOfMyStruct.map { obj in
var obj = obj
obj.backgroundColor = .red
return obj
}
It can be simplified by adding this Array extension.
Swift 4
extension Array {
mutating func mutateEach(by transform: (inout Element) throws -> Void) rethrows {
self = try map { el in
var el = el
try transform(&el)
return el
}
}
}
Usage
arrayOfMyStruct.mutateEach { obj in
obj.backgroundColor = .red
}
Solution 4
For Swift 3, use the enumerated()
method.
For example:
for (index, _) in arrayOfMyStruct.enumerated() {
arrayOfMyStruct[index].backgroundColor = UIColor.redColor()
}
The tuple also includes a copy of the object, so you could use for (index, object)
instead to get to the object directly, but since it's a copy you would not be able to mutate the array in this way, and should use the index
to do so. To directly quote the documentation:
If you need the integer index of each item as well as its value, use the enumerated() method to iterate over the array instead. For each item in the array, the enumerated() method returns a tuple composed of an integer and the item.
Avba
Updated on August 30, 2021Comments
-
Avba over 2 years
I am still not sure about the rules of struct copy or reference.
I want to mutate a struct object while iterating on it from an array: For instance in this case I would like to change the background color but the compiler is yelling at me
struct Options { var backgroundColor = UIColor.blackColor() } var arrayOfMyStruct = [MyStruct] ... for obj in arrayOfMyStruct { obj.backgroundColor = UIColor.redColor() // ! get an error }
-
Entalpi about 8 yearsThat 'var' makes the item you are iterating over locally mutable (but does not change the backing struct).
-
dudeman about 8 yearsI have one clarifying question: If you had specified a name for the current
MyStruct
object and since the array is passed by value, trying to print out the named struct object'sbackgroundColor
on the next line would result in the old value being printed, correct? -
ldoogy over 7 yearsThis does not work for Swift 3, please see my answer below for Swift 3.
-
Mihai Erős almost 6 yearsBeware of the discussion from Apple's docs:
A collection’s indices property can hold a strong reference to the collection itself, causing the collection to be nonuniquely referenced. If you mutate the collection while iterating over its indices, a strong reference can result in an unexpected copy of the collection. To avoid the unexpected copy, use the index(after:) method starting with startIndex to produce indices instead.
Source: [developer.apple.com/documentation/swift/collection/… -
jvarela over 4 yearsThe link in @Mihai Erös comment was moved to: developer.apple.com/documentation/swift/string/2949967-indices
-
Peter Schorn almost 4 yearsThere's no need to use the enumerated method if you only need the indices. Just use
for i in 0..< arrayOfMyStruct.count
-
cumanzor over 3 years@MihaiErős would this apply to
for (index, _) in arrayOfMyStruct.enumerated()
as well? -
Ralf Ebert about 3 yearsCorrect link for the documentation page: developer.apple.com/documentation/swift/collection/…
-
Ralf Ebert about 3 yearsThe enumerated() docs do not mention anything about that. Also I'd still use .indices just for the sake of clarity; but good to know to pay attention if you're handling a very large list.