Thursday, May 7, 2015

Phân biệt những kiểu khai báo file #include, #import và @import trong Objective C (Có hình ảnh minh hoạ)

Trong ngôn ngữ lập trình Objective C, khi khởi tạo 1 đối tượng hệ thống sẽ thường sinh ra 2 file có đuôi khác nhau là file ".h" và file ".m". Vậy câu hỏi được đặt ra là khi ta khai báo file nên khái báo trong file .h hay file .m ? Chúng có sự khác biệt nào không? Hãy cùng mình phân tích sự khác biệt khi khai báo trong từng loại file.

Ví dụ: Mình tạo 3 đối tượng có tên như: ObjectA, ObjectB, ObjectC. Trong ObjectA mình có dùng ObjectB, trong ObjectB mình có dùng ObjectC và trong ObjectC mình có dùng ObjectA. Những đối tượng này đều có thuộc tính là "name".

- Trường hợp 1: Giả sử mình khai báo bằng #include trong những file .h. Như hình sau:


Khi build thì XCode báo thành công, nhưng khi ta #include Object A trong file ViewController.h thì sau khi build XCode sẽ báo lỗi như: "fatal error: too many errors emitted, stopping now [-ferror-limit=]"

Khi XCode báo lỗi này có nghĩa chúng ta đã khai báo những đối tượng này chồng lên nhau vì thế không thể nào build thành công.
Hình ảnh minh hoạ:

- Trường hợp 2: Chúng ta sẽ thay thế bằng cách dùng #import thay cho #include. Sau khi build XCode sẽ báo thành công.
Chúng ta đã sửa được lỗi khai báo vòng? Xin thưa là vẫn chưa fix được lỗi khai báo vòng theo kiểu này vì nếu giả sử ta cũng phải khai báo đối tượng ViewController trong ObjectA  và khai báo ObjectB trong ViewController, thì sau khi build XCode cũng sẽ báo lỗi khai báo vòng.
Hình ảnh minh hoạ:
Làm thế nào chúng ta giải quyết vấn đề này? Cách giải quyết như sau: Chúng ta nên dùng từ khoá là @class để khai báo đối tượng đó trong file .h và #import đối tượng đó trong file .m.


Kết luận:
- Luôn khai báo đối tượng trong file .m và dùng từ khoá @class để khai báo đối tượng trong file .h.
- Ngoại trừ những trường đặc biệt mới khai báo trong file .h:
  + Ví dụ: khai báo custom delegate của những đối tượng trong file .h thì nên khai báo trong file .h.
  + Ví dụ 2: Khai báo quan hệ cha và con trong Order và Order detail. Nếu trong Order.h ta có import OrderDetail.h thì khi sử dụng trong những file khác ta chỉ cần import Order.h thì ta vẫn sử dụng được đối tượng OrderDetail.

Phân biệt những kiểu khai báo trong Objective C:
- #include: thường được dùng để khai báo những đối tượng viết bằng ngôn ngữ C/C++.
- #import: dùng để khai báo những đối tượng viết bằng ngôn ngữ Objective C.
- @import: cũng giống như #include và #import nhưng dùng để khai báo đối tượng thông qua cơ chế Modules. Chỉ hỗ trợ từ XCode 5 và iOS 7 trở lên. Điểm mạnh của cách khai báo này là ta chỉ cần khai báo những đối tượng trong hệ thống mà không cần phải add framework đó vào project. Hệ thống sẽ tự động tìm đối tượng đó trong những framework của system để import vào. Tham khảo thêm trong tài liệu tham khảo [3].
Khai báo kiểu bình thường:
 #import "CoreAudio.h"  
Sẽ khai báo bằng cách:
 @import CoreAudio;  

Công cụ dùng để viết: XCode 6.3, iOS 8.3

Tài liệu tham khảo:
- [1] http://stackoverflow.com/questions/13250253/difference-between-includes-and-imports
- [2] http://stackoverflow.com/questions/1044360/import-using-angle-brackets-and-quote-marks
- [3] http://stackoverflow.com/questions/5425465/does-objective-c-allow-circular-dependencies

No comments:

Post a Comment

Note: Only a member of this blog may post a comment.