Removing object from NSMutableArray

46,762

Solution 1

You cannot remove objects from array while fast-enumerating it:

numeration is “safe”—the enumerator has a mutation guard so that if you attempt to modify the collection during enumeration, an exception is raised.

Anyway why do you need to change you container while enumerating it? Consider storing elements that need to be deleted and remove them from your container using removeObjectsInArray: or removeObjectsAtIndexes: method.

Solution 2

Just add keyword break; after removing the item...

for(id item in items) {
    if([item isEqual:itemToDelete]) {
        [items removeObject:item];
        break; // A very important line 🛑
    }
}

Solution 3

An Objective-C collection must not be modified during enumeration.

You may use this variant to delete objects from collection:

for (NSInteger i = items.count - 1; i >= 0 ; i--) {
   [items removeObjectAtIndex:i];
}

Solution 4

The former loop is a "for-each" loop in Objective C.

*i is a pointer to the direct item in the items-Array (most of the time this will be NSMutableArray).

This way you can operate directly on the item:

[items removeObject: i];

This (should) work - I am currently not working on my Mac and can't check it. However it might be that Objective-C Prevents removing objects while iterating over the collection (that is quite common in most languages).

Share:
46,762
Ben Packard
Author by

Ben Packard

iOS developer and program manager located in Washington, DC.

Updated on September 28, 2021

Comments

  • Ben Packard
    Ben Packard over 2 years

    I stumbled across the following shortcut in setting up a for loop (shortcut compared to the textbook examples I have been using):

    for (Item *i in items){ ... }
    

    As opposed to the longer format:

    for (NSInteger i = 0; i < [items count]; i++){ ... } //think that's right
    

    If I'm using the shorter version, is there a way to remove the item currently being iterated over (ie 'i')? Or do I need to use the longer format?

  • Nick Forge
    Nick Forge about 14 years
    It is indeed the case that you can't remove (or mutate) an NSMutableArray while enumerating over it, so the removeObject: call will raise an exception. (as explained by Vladimir in his answer)
  • Ben Packard
    Ben Packard about 14 years
    Thanks. I am changing it while enumerating it because it is a temporary queue. I'm moving all the items to another array.
  • JeremyP
    JeremyP almost 14 years
    If you are moving all the objects to another array, try the following: [destinationArray addObjectsFromArray: sourceArray]; [sourceArray removeAllObjects];
  • Groot
    Groot over 11 years
    Great answer! Works like a charm!
  • rptwsthi
    rptwsthi almost 11 years
    @Abdullah71 with id you never use *.
  • marsbear
    marsbear over 7 years
    This will remove one object and stop iterating, thus missing possible other objects that should be removed. So only use this if you need to remove zero to one object per iteration.
  • marsbear
    marsbear over 7 years
    Only until i gets -1... then it crashes :)
  • marsbear
    marsbear over 7 years
    Imho the shortest way to do it properly, given that it's fine with your business logic to traverse the array from back to front.