Swift - How to mutate a struct object when iterating over it

20,667

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.

Share:
20,667
Avba
Author by

Avba

Updated on August 30, 2021

Comments

  • Avba
    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
    Entalpi about 8 years
    That 'var' makes the item you are iterating over locally mutable (but does not change the backing struct).
  • dudeman
    dudeman about 8 years
    I 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's backgroundColor on the next line would result in the old value being printed, correct?
  • ldoogy
    ldoogy over 7 years
    This does not work for Swift 3, please see my answer below for Swift 3.
  • Mihai Erős
    Mihai Erős almost 6 years
    Beware 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
    jvarela over 4 years
    The link in @Mihai Erös comment was moved to: developer.apple.com/documentation/swift/string/2949967-indic‌​es
  • Peter Schorn
    Peter Schorn almost 4 years
    There's no need to use the enumerated method if you only need the indices. Just use for i in 0..< arrayOfMyStruct.count
  • cumanzor
    cumanzor over 3 years
    @MihaiErős would this apply to for (index, _) in arrayOfMyStruct.enumerated() as well?
  • Ralf Ebert
    Ralf Ebert about 3 years
    Correct link for the documentation page: developer.apple.com/documentation/swift/collection/…
  • Ralf Ebert
    Ralf Ebert about 3 years
    The 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.