SoFunction
Updated on 2025-04-04

Detailed explanation of MRC (Manual Memory Management) for iOS development

Preface:

In iOS, use reference counting to manage OC object memory

The default reference count of a newly created OC object is 1. When the reference count is reduced to 0, the OC object will be destroyed, freeing up the memory space it consumes.

Calling retain will cause the OC object to count +1, and calling release will cause the OC object to count -1.

Summary of experience in memory management

When calling alloc, new, copy, and mutableCopy methods, returns an object. When this object is not needed, release or autorelease must be called to release it.
If you want to have an object, let its reference count +1; if you don't want to have an object, let its reference count -1.

1. MRC manually manages memory (Manual Reference Counting)

1. Reference counter

Reference counter:

An integer, represented as "number of times the object is referenced". The system needs to determine whether the object needs to be recycled based on the object's reference counter.

Regarding the "reference counter", there are the following characteristics:

  • Each OC object has its own reference counter.
  • When any object is first created, the initial reference count is 1.
  • Even when creating an object using alloc, new or copy, the object's reference counter is 1 by default.
  • The system will recycle this object when no one uses it. That is to say:
  • When the object's reference counter is 0, the memory occupied by the object will be recycled by the system.
  • If the object's reference counter is not 0, then the memory it consumes cannot be recycled during the entire program running process (unless the entire program has exited).

2. Reference counter operation

  • To ensure the existence of an object, whenever a reference is created to the object, you need to send a retain message to the object, the reference counter value +1 can be made (the retain method returns to the object itself).
  • When an object is no longer needed, the reference counter value -1 can be made by sending a release message to the object.
  • Send a retainCount message to the object to get the current reference counter value.
  • When the object's reference count is 0, the system knows that the object no longer needs to be used, so it can free its memory and initiate this process by sending a dealloc message to the object.
  • It should be noted that release does not mean destroying/recycling objects, it is just to convert counter -1.
// Create an object, the default reference counter is 1RHPerson *person1 = [[RHPerson alloc] init];
NSLog(@"retainCount = %zd", [person1 retainCount]);

// Just send a retain message to the object, and add 1 to the reference counter[person1 retain];
NSLog(@"retainCount = %zd", [person1 retainCount]);

// Just send a release message to the object, and the reference counter is reduced by 1[person1 release];
NSLog(@"retainCount = %zd", [person1 retainCount]);

// When retainCount is equal to 0, the object is destroyed[person1 release];

NSLog(@"--------------");
2022-07-11 16:09:24.102850+0800 Interview01-Memory management[8035:264221] retainCount = 1
2022-07-11 16:09:24.103083+0800 Interview01-Memory management[8035:264221] retainCount = 2
2022-07-11 16:09:24.103126+0800 Interview01-Memory management[8035:264221] retainCount = 1
2022-07-11 16:09:24.103231+0800 Interview01-Memory management[8035:264221] -[RHPerson dealloc]
2022-07-11 16:09:24.103259+0800 Interview01-Memory management[8035:264221] --------------
Program ended with exit code: 0

3. Dealloc method

  • When the reference counter value of an object is 0, the object will be destroyed and the memory it occupies is recycled by the system.
  • When the object is about to be destroyed, the system will automatically send a dealloc message to the object (so, whether the dealloc method has been called can be determined whether the object has been destroyed)
  • Rewriting of the dealloc method (note that it is in MRC)

The dealloc method is generally rewritten, and relevant resources are released here. dealloc is the last word of the object.
Once the dealloc method is rewritten, [super dealloc] must be called and placed at the end.

- (void)dealloc {
    NSLog(@"%s", __func__);
    [super dealloc];
}

Notes on using dealloc:

The dealloc method cannot be called directly.

Once the object is recycled, the memory it consumes is no longer available, and persisting in using it will cause the program to crash (wild pointer error).

4. Wild pointers and empty pointers

As long as an object is released, we call this object a "zombie object (an object that cannot be used anymore).

When a pointer points to a zombie object (an object that cannot be used anymore), we call this pointer a "wild pointer".

An error will be reported as long as a message is sent to a wild pointer (EXC_BAD_ACCESS error).

RHPerson *person1 = [[RHPerson alloc] init];
[person1 release];
[person1 release];
[person1 release];

In order to avoid errors when sending messages to wild pointers, generally, when an object is released, we will set the pointer of this object to a null pointer.

Null pointer:

There is no pointer to the storage space (nil is stored, that is, 0).

Sending a message to a null pointer is not a response.

RHPerson *person1 = [[RHPerson alloc] init];
[person1 release];
person1 = nil;
[person1 release];

2. Memory management idea

1. Single object memory management idea

Thought 1: The object you create, hold it yourself, and be responsible for releasing it yourself

Create and hold objects through the alloc, new, copy, or mutableCopy methods.
When the object you hold is no longer needed, the release or autorelease method must be called to release the object.

id obj1 = [[NSObject alloc] init];
[obj1 release];

id obj2 = [NSObject new];
[obj2 release];

Thought 2: You can hold the object you don’t create by yourself

In addition to the object obtained using the above method (alloc / new / copyright / mutableCopy method), it is not the holder of the object itself because it is not generated and held by itself.
By calling the retain method, you can hold the object yourself even if it is not an object created by yourself.
Similarly, when the object you hold is no longer needed, the release method must be called to release the object.

id obj3 = [NSArray array];
[obj3 retain];
[obj3 release];

Whether it is an object created by you or not, you can hold it yourself and be responsible for release.
There is an increase or decrease in counter.
If you once had an object's counter +1, you must have an object counter -1 at the end.

2. Multiple object memory management ideas

Multiple objects are often connected through the setter method, and their memory management methods are also implemented in the setter method and the dealloc method. So only by understanding how the setter method is implemented can we understand the memory management idea between multiple objects.

#import <Foundation/>

#import ""

NS_ASSUME_NONNULL_BEGIN

@interface RHPerson : NSObject

{
    RHRoom *_room;
}

- (void)setRoom:(RHRoom *)room;

- (RHRoom *)room;
@end

NS_ASSUME_NONNULL_END
#import ""

@implementation RHPerson

- (void)setRoom:(RHRoom *)room {
    if (_room != room) {
        [_room release];
        _room = [room retain];
    }
}

- (RHRoom *)room {
    return _room;
}

- (void)dealloc {
    [_room release];
    [super dealloc];
    NSLog(@"%s", __func__);
}

@end

3. @property parameters

Add @property before the member variable, and the system will automatically generate the basic setter/getter method, but will not generate memory management-related code.
@property(nonatomic) int val;
Similarly, if you add assign to the property, the system will not help us generate the setter method memory management code. It will only generate ordinary getter/setter methods. By default, nothing is written is assign.
@property(nonatomic, assign) int val;
If you add retain after the property, the system will automatically generate the memory management code of the getter/setter method, but we still need to rewrite the dealloc method ourselves.
@property(nonatomic, retain) RHRoom *room;

4. Automatic release pool

1. Automatic release pool

When we no longer use an object, it should be freed, but sometimes we don't know when it should be freed. To solve this problem, Objective-C provides the autorelease method.

autorelease is a memory management method that supports reference counting. As long as an autorelease message is sent to an object, the object will be placed in an automatic release pool. When the automatic release pool is destroyed, a release operation will be performed on "all objects" in the pool.
Note: This is just a release message. If the reference-counted at that time is still not 0, the object will not be released.

The autorelease method returns the object itself, and after the autorelease method is called, the object's counter remains unchanged.

NSObject *obj = [NSObject new];
[obj autorelease];
NSLog(@" = %zd", );

2. What are the benefits of using autorelease?

No need to care about the time of the object being released
No need to care about when to call release

3. What is the principle of autorelease in essence?

​ autorelease actually just delays the call to release. For each autorelease, the system simply puts the object into the current autorelease pool. When the pool is released, all objects in the pool will be called the release method.

4. How to create autorelease

// The first method: create using NSAutoreleasePool// Create an automatic release poolNSAutoreleasePool *pool = [[NSAutoreleasePool alloc] init];
// Destroy the automatic release pool[pool release];
// The second way: create using @autoreleasepool@autoreleasepool {
// Start to create an automatic release pool// End means destroying the automatic release pool}

5. How to use autorelease

NSAutoreleasePool *pool = [[NSAutoreleasePool alloc] init];
RHPerson *p = [[[RHPerson alloc] init] autorelease];
[pool release];
@autoreleasepool {
// Start to create an automatic release poolRHPerson *p = [[[RHPerson alloc] init] autorelease];
// End means destroying the automatic release pool}

6. Notes on autorelease

It is not automatically added to the automatic release pool code.

@autoreleasepool {
// Because autorelease is not called, it is not added to the automatic release poolRHPerson *p = [[RHPerson alloc] init];
// End means destroying the automatic release pool}

Send autorelease outside the autorelease pool will not be added to the autorelease pool
autorelease is a method that is only valid if called in the autorelease pool.

@autoreleasepool {
}
// There is no corresponding automatic release pool. Only when autorelease is called in the automatic release pool will it be placed in the free pool.Person *p = [[[Person alloc] init] autorelease];
[p run];
 
// Correct writing@autoreleasepool {
    Person *p = [[[Person alloc] init] autorelease];
 }
 
// Correct writingPerson *p = [[Person alloc] init];
@autoreleasepool {
    [p autorelease];
}

7. Automatically release nested use of pools

The automatic release pool exists in the form of a stack.
Since the stack has only one entry, calling autorelease will place the object to the automatic release pool at the top of the stack.
The top of the stack is the automatic release pool closest to calling the autorelease method.

@autoreleasepool { // Automatic release pool at the bottom of the stack    @autoreleasepool {
        @autoreleasepool { // Automatic release pool on top of the stack            Person *p = [[[Person alloc] init] autorelease];
        }
        Person *p = [[[Person alloc] init] autorelease];
    }
}

Automatically release the pool and it is not suitable to place objects that occupy more memory.
Try to avoid using this method on large memory, and try to use this delayed release mechanism as little as possible.
Do not put a large number of loop operations between the same @autoreleasepool, as this will cause a memory peak to rise.

// Memory surges@autoreleasepool {
    for (int i = 0; i &lt; 99999; ++i) {
        Person *p = [[[Person alloc] init] autorelease];
    }
}
// Memory will not surgefor (int i = 0; i &lt; 99999; ++i) {
    @autoreleasepool {
        Person *p = [[[Person alloc] init] autorelease];
    }
}

8. Autorelease Error Usage

Do not call autorelease continuously.

@autoreleasepool {
 // Wrong writing, excessive release    Person *p = [[[[Person alloc] init] autorelease] autorelease];
 }

After calling autorelease, release is called (error).

@autoreleasepool {
    Person *p = [[[Person alloc] init] autorelease];
    [p release]; // Wrong writing, excessive release}

5. Avoid circular references in MRC

Define two classes Person class and Dog class

Person class:

#import <Foundation/>
@class Dog;
 
@interface Person : NSObject
@property(nonatomic, retain)Dog *dog;
@end

Dog class:

#import <Foundation/>
@class Person;
 
@interface Dog : NSObject
@property(nonatomic, retain)Person *owner;
@end

Execute the following code:

int main(int argc, const char * argv[]) {
    Person *p = [Person new];
    Dog *d = [Dog new];
 
     = d; // retain
     = p; // retain  assign
 
    [p release];
    [d release];
 
    return 0;
}

It will appear that object A must have object B, and B must have object A, which will form a loop retain, resulting in object A and object B that will never be released.

So how to solve this problem?

Don't let A retain B, B retain A.
Just let one of the parties not do the retain operation.
When both ends refer to each other, one end should use retain and the other end should use assign.

This is the end of this article about MRC (Manual Memory Management) in iOS development. For more related MRC manual memory management content, please search for my previous articles or continue browsing the related articles below. I hope everyone will support me in the future!