tags:

views:

55

answers:

3

Can someone explain to me what init and alloc do in Obj-C. I am reading this obj-c book that gives an example of creating object but it does not really go into details of what it does. What does alloc return? what does init return?

Animal * k = [Animal alloc];
k = [k init];
+1  A: 

In its simplest form:

  • alloc: short for allocation, reservers a memory location and returns the pointer to that memory location. This pointer is then stored in the k variable.

  • init: short for initialization, sets up the object and returns the object. The init method depends on the object, but it generally involves sending the init message to the superclass. And if that init method (of the superclass) returns an object (not nil) some ivars may be set up depending on the task of that class.

--

Example of an init implementation, the Schedule class initializes its channels ivar with an empty array. Basically your giving the Schedule object a chance to sort itself out, so it can start receiving messages once it is created. You could remove the channels initialization from the init method, but then you would have to check if the channels ivar is nil in every method of the Schedule class and initialize it if it is indeed nil.

- (Schedule*)init {
  self = [super init];

  if (self) {
    channels = [[NSMutableArray alloc] initWithCapacity:0];
  }

  return self;
}
Yannick Compernol
what does setting up the object mean? what do they return?
denniss
I've extended my answers a little bit.
Yannick Compernol
+1  A: 

alloc and init are two methods that are inherited from NSObject you can provide your own methods or call the ones from NSObject

alloc allocates the memory to create a new instance of your class(@interface) init initializes the contents of that instance, by default init sets all member values to 0/nil however you can create your own init method to customize what is done.

Animal * k = [[Animal alloc] init]; // creates a new Animal object

you can also write

Animal * k = [Animal new]; // which would be a bit more similar to other languages
Anders K.
+2  A: 
  • alloc allocates a chunk of memory to hold the object, and returns the pointer.

    MyClass*myObj=[MyClass alloc];
    

    myObj cannot be used yet, because its internal state is not correctly set up. So, don't write a code like this.

  • init sets up the initial condition of the object and returns it. Note that what's returned by [a init] might be different from a. That explains the code Yannick wrote:

    -init{
         self=[super init]; // 1.
         if(self){          // 2.
             ....
         }
         return self;       // 3.
    }
    
    1. First, you need to call the superclass's init, to set up the superclass's instance variables, etc. That might return something not equal to the original self, so you need to assign what's returned to self.
    2. If self is non-nil, it means the part controlled by the superclass is correctly initialized. Now you perform your initialization. All of the instance variables are set to nil (if it's object) and 0 if it's integer. You'll need to perform additional initial settings.
    3. Return the set-up self. The returned self might be different from what's allocated! So, you need to assign the result of init to your variable.

This gives us an important lesson: never split the call to alloc and init. Don't do

 MyClass*myObj = [MyClass alloc];
 [myObj init];

because [myObj init] might return something else. Don't do

 MyClass*myObj = [MyClass alloc];
 myObj=[myObj init];

because you will eventually forget to write the part myObj= in the second line.

Always do

 MyClass*myObj = [[MyClass alloc] init];

I don't recommend

 MyClass*myObj = [MyClass new];

either, because it does not correctly call the initialization method: some class doesn't accept a plain init. For example, NSView needs initWithFrame:, which can't be called with new. So, don't use new.

Yuji