The difference between 'copy' and 'mutableCopy' can be simply understood with polymorphism in Object Oriented Programming concepts.
We will take the example of Array in objective-C. MutableArray is the extension of NSArray class. Therefore, all the methods available in NSArray is available in NSMutableArray, but the additional methods present in NSMutableArray is not known to NSArray class.
Now moving ahead, the copy method on an NSArray will return an object of type NSArray(The array that can not be modified). And mutableCopy method will return an object of mutable type (The array that can be modified).
Now, there can below cases :
Case 1:
NSArray *arr1=[NSArray arrayWithObjects:@"A",@"B",@"C", nil];
NSArray *arr2=[arr1 copy];
NSLog(@"arr1:%@",[arr1 description]);
NSLog(@"arr2:%@",[arr2 description]);
In this case an NSArray object is returned and is received in an NSArray object.
So, the array received can not be modified.
Therefore, we will not be able to use below statement
[arr2 insertObject:@"Z" atIndex:0];
Case 2:
NSArray *arr1=[NSArray arrayWithObjects:@"A",@"B",@"C", nil];
NSArray *arr2=[arr1 mutableCopy];
NSLog(@"arr1:%@",[arr1 description]);
NSLog(@"arr2:%@",[arr2 description]);
In this case an NSMutableArray object is returned and is received in an NSArray object.
Since the receiver object is of type NSArray, it doesn't know the methods present in NSMutableArray, arr2 will not be able to use any of the methods of NSMutableArray.
That is, the method mutableArray will make no sense in this scenario.
So, we will not be able to use below statement
[arr2 insertObject:@"Z" atIndex:0];
Case 3:
NSArray *arr1=[NSArray arrayWithObjects:@"A",@"B",@"C", nil];
NSMutableArray *arr2=[arr1 copy];
NSLog(@"arr1:%@",[arr1 description]);
NSLog(@"arr2:%@",[arr2 description]);
In this case an NSArray object is returned and is received in an NSMutableArray type object. The receiver arr2 is now pointing to an object address that is of type NSArray. However arr2 has the additional methods than NSArray, it will not be able to use those methods coz the pointed object NSArray does not know the additional methods present in arr2(NSMutableArray).
Hence, we will not be able to use below statement
[arr2 insertObject:@"Z" atIndex:0];
Case 4:
NSArray *arr1=[NSArray arrayWithObjects:@"A",@"B",@"C", nil];
NSMutableArray *arr2=[arr1 mutableCopy];
NSLog(@"arr1:%@",[arr1 description]);
NSLog(@"arr2:%@",[arr2 description]);
[arr2 insertObject:@"Z" atIndex:0];
In this scenario, the receiver(arr2) of type NSMutableArray receives an object of type NSMutableArray. Therefore, the receiver knows the additional methods of NSMutableArray as well as the object that is being pointed by receiver(arr2).
And finally below statement will work like charm,
[arr2 insertObject:@"Z" atIndex:0];
Log before insertion will be:
arr1:(
A,
B,
C
)
arr2:(
A,
B,
C
)
And log after insertion will be:
arr1:(
A,
B,
C
)
arr2:(
Z,
A,
B,
C
)
Hope above description helped you.