Tuesday, July 2, 2013

iOS: Ngôn ngữ lập trình Objective C - P3



1.1.                     Quản lý bộ nhớ

1.1.1.     Các nguyên tắc quản lý bộ nhớ

Nguyên tắc cơ bản
·        Khi bạn nắm quyền sở hữu một đối tượng, khởi tạo đối tượng bằng các phương thức mà trong tên bắt đầu với với alloc hoặc new hoặc copy (ví dụ, alloc, newObject hoặc mutableCopy…) hoặc gửi một thông điệp retain, bạn phải có trách nhiệm giải phóng quyền sở hữu đối tượng đó bằng cách sử dụng release hoặc autorelease. Bất kỳ khi nào bạn nhận được một đối tượng (không phải tự mình khởi tạo), bạn không được release nó.

Các nguyên tắc khác
·        Khi bạn cần lưu trữ một đối tượng được nhận như một propertytrong một biến thể hiện, bạn phải retain hoặc copy nó. (Điều này không đúng cho cho tham khảo yếu, nhưng đây là điển hình hiếm).
·        Một đối tượng được nhận thường đảm bảo vẫn có hiệu lực trong phương thức mà nó đã được nhận (ngoại trừ trong các ứng đa luồng và vài trường hợp Distributes Objects). Phương thức đó có thể trả về an toàn đối tượng mà nó được triệu gọi. Sử dụng retain trong việc kết hợp với release hoặc autorelease khi cần thiết để bảo vệ một đối tượng khỏi hiệu lực của các thông điệp không hợp lệ bên ngoài.
·        autoreleasecó nghĩa là “gửi một thông điệp releasesau đó


1.1.2.     Vấn đề khi khởi tạo đối tượng

Xét đoạn code khởi tạo đối tượng sau

  TestClass *ptr = [TestClass alloc];
  [ptr init];

  // Do something with the object
  if (ptr)
    ...
Thoạt nhìn qua có vẻ vô hại, phương thức alloc khởi tạo một vùng nhớ cho đối tượng ptr, sau đó gửi thông điệp gọi phương thức init cho đối tượng ptr, lỗi có thể xảy ra ở đây. Giả sử dòng lệnh ở dòng lệnh [ptr init], phương thức init có lỗi xảy ra và trả về nil, tiếp theo ở dòng lệnh if, lúc này ptr vẫn khác nil mặc dù phương thức inittrả về nil (nhưng nó không được gán cho ptr), dẫn tới ý nghĩa của phương thức tạo thực sự không còn nữa.

Khi đó, đoạn code sau sẽ giải quyết được khả năng lỗi ở trên

  TestClass *ptr = [[TestClass alloc] init];

  // Do something with the object
  if (ptr)
    ...

Giả sử initcó lỗi và trả về nilthì ptr vẫn sẽ được gán là nil, do đó dòng lệnh ifsẽ thực hiện đúng ý nghĩa của nó.
Ý nghĩa từ ví dụ này đó là, ta luôn luôn phải trả về nil trong phương thức initnếu có lỗi khởi tạo xảy ra và lưu ý phải kết hợp 2 lời gọi phương thức allocinit.

1.1.3.     Release

Nếu bạn khởi tạo một đối tượng sử dụng cách thủ công alloc, bạn cần phải release đối tượng sau đó. Bạn không nên thực hiện release thủ công một đối tượng autorealse bởi vì ứng dụng có thể sẽ bị crash nếu bạn làm điều đó.

            // string1 will be released automatically
          NSString* string1 = [NSString string];

          // must release this when done
          NSString* string2 = [[NSString alloc] init];
          …..
          [string2 release];

1.1.4.     Retain


Trong Objective-C, mỗi đối tượng có một bộ đếm được sử dụng để kiểm soát tất cả các tham chiếu bởi đối tượng hoặc nó có.
Để biết được giá trị của bộ đếm này ta sử dụng thuộc tính retainCount [object retainCount]. Phương thức alloc, new, copyretain đều tăng bộ đếm này lên 1 và phương thức release giảm bộ đếm này đi 1, khi bộ đếm có giá trị bằng 0 thì phương thức dealloc của đối tượng sẽ được gọi.



Bất cứ khi nào một đối tượng có nhu cầu được sử dụng bởi một đối tượng khác nó phải retain để tăng bộ đếm retainCount lên 1, và khi nó không còn sử dụng nữa thì phải release để giảm bộ đếm retainCount đi 1. Khi bộ đếm có giá trị bằng 0 có nghĩa là nó không còn nhu cầu để sử dụng nữa, nó sẽ tự hủy bằng phương thức dealloc.

1.1.5.      Dealloc

Phương thức dealloc được gọi khi đối tượng đang được remove khỏi bộ nhớ. Nó thường được sử dụng nhất khi giải phóng tất cả các tham chiếu của các biến thể hiện con của đối tượng khỏi bộ nhớ. Hay nói cách khác, một lớp nếu có các biến thể hiện là các đối tượng thì trong phương thức dealloc của lớp phải thực hiện giải phóng các biến thể hiện này.

- (void)dealloc {
          [childVar1 release];
          [childVar2 release];
          [super dealloc];
}
Phương thức [super dealloc] được sử dụng để thông báo cho lớp cha thực hiện việc “dọn dẹp” (đây là phương thức được định nghĩa trong NSObject), nếu không gọi phương thức này, có thể đối tượng sẽ không được remove khỏi bộ nhớ, sẽ gây nên tình trạng chiếm bộ nhớ.

#import
#import "Address.h"

@interface Person : NSObject {
          Address* address;
}
-(Person*) constructor: (Address*) add;
-(void) show;
@end

#import "Person.h"

@implementation Person
-(Person*) constructor: (Address*) add{
          self = [super init];
          if(self){
                   address = add;
          }
          return self;
}
-(void) show{
          printf("Person: \n");
          printf("   --- ");
          [address show];
}
-(void) dealloc{
          printf("Dealloc method of Person\n");
          [address release];
          [super dealloc];
}
@end


#import
#import "Person.h"
#import "Name.h"

@interface VNPerson : Person {
          Name* name;
}
-(VNPerson*) constructor: (Address*) add : (int) idn;
@end

#import "VNPerson.h"
#import "Name.h"

@implementation VNPerson

-(VNPerson*) constructor: (Address*) add : (int) idn{
          self = [super constructor: add];
          if(self){
                   name = [[Name alloc] constructor: idn];
          }
          return self;
}
-(void) show{
          [super show];
          printf("   --- ");
          [name show];
}
-(void) dealloc{
          printf("Dealloc method of VNPerson\n");
          [name release];
          [super dealloc];// Mục đích giải phóng biến thể hiện address trong lớp cha
}
@end


#import "Address.h"
#import "VietnamAddress.h"
#import "Person.h"
#import "VNPerson.h"

int main(int argc, char *argv[]) {
   
          Address *address = [[Address alloc] constructor: 01 : 02];
          [address show];
          VietnamAddress *vnaddress = [[VietnamAddress alloc] constructor: 03 : 04];
          [vnaddress show];
          [vnaddress changeAddress: 05];
         
          Person *person = [[VNPerson alloc] constructor: address : 1915];
          [person show];
         
          [address release];
          [vnaddress release];
          [person release];
                  
          return 1;
}

Lưu ý: Phương thức dealloc sẽ không được gọi trong trường hợp bộ dọn rác được bật.





1.1.6.     Tham chiếu yếu

Retain một đối tượng tạo ra một tham chiếu “mạnh” đến đối tượng đó. Một đối tượng không thể dealloc cho tới khi tất cả các tham chiếu mạng được giải phóng hết.

Trong một số trường hợp, ta có thể muốn có một tham chiếu đến đối tượng mà không cản trở việc đối tượng tự giải phóng chính nó, lúc này ta có thể thiết lập một tham chiếu “yếu” đến đối tượng.
Một tham chiếu yếu được tạo ra bằng cách chứa một con trỏ trỏ đến một đối tượng mà không retain đối tượng đó. Tham chiếu yếu rất cần thiết trong việc thiết lập các tham chiếu vòng. Ví dụ, đối tượng A và đối tượng B cần trao đổi thông tin với nhau nên mỗi đối tượng cần một tham chiếu đến đối tượng kia (ví như mới quan hệ giữa 2 đối tượng cha - con), nếu như chúng ta retain đối tượng kia khi thiết lập tham chiếu thì mỗi đối tượng chỉ được dealloc khi nào kết nối này đứt, tuy nhiên kết nối này chỉ đứt khi có một đối tượng được dealloc mà thôi. Trong trường hợp này ta thấy được lợi ích của tham chiếu yếu.
Chúng ta phải thật cẩn thận khi truyền thông điệp trong tham chiếu yếu, trong trường hợp đối tượng nhận thông điệp đã dealloc thì ứng dụng có thể sẽ bị crash. Đồng thời, trong mối quan hệ tham chiếu yếu, đối tượng được tham chiếu đến phải có trách nhiệm báo cho đối tượng kia biết khi nó thực hiện dealloc, ví dụ gửi một thông điệp setDelegate:với tham số nil cho đối tượng kia chẳng hạn, trong trường hợp đối tượng dealloc là một Delegate của đối tượng nhận thông điệp.

No comments:

Post a Comment