Effective Objective-C(一)

最近在看Effective Objective-C,顺便做个笔记。

Familiarize Yourself with Objective-C's Root
Objective-C由Smaltalk演化而来,Smaltalk是消息传递机制的鼻祖。因此OC也采用消息传递机制取代功能调用(function calling)。
消息传递与功能调用的区别如下:

//Messaging(OC)  
Object *obj = [Object new];  
[obj performWith:para1 and:para2];

//function calling(C++)
Object *obj = new Object;
obj->perform(para1,para2);

注意区别在于:使用消息结构的语言,运行时所执行的代码由运行环境来决定,而使用含住调用的语言,择优编译器决定。如果代码调用的函数是多态的,那么在运行时要按照“虚方法表(virtual table)”来查出到底该执行哪个函数实现,而采用消息结构的语言,不论是否多态,总是在运行时才回去查找所要执行的方法。甚至编译器对于接收消息是何种类型也不会关心。接收消息的对象问题也要在运行时处理,这一过程叫做“动态绑定(dynamic binding)”。

Objective-C的重要工作都由“运行期组件而非编译器来完成”,使用Objective-C的面向对象特性所需的全部数据结构以及函数都在运行期组件里面。运行期组件本质上是一种与开发者所编代码相链接的“动态库”,其代码能把开发者所编写的所有程序粘合起来
Objective-C对象的声明:NSString *someString = @"The string";解释为生命了一个名为someString 的变量,其类型是NSString *,也就是说变量为指向NSString的指针。

所有Objective-C对象所占内存都是分配在“堆空间”中,而绝不会分配在“栈”上。
在Objective-C中,有时会遇到定义中不含 * 的变量,它们可能会使用“栈控件”,这些变量保存的不是Objective-C对象。比如CoreGraphics框架中的CGRect:

CGRect frame;
frame.origin.x = 0.0f;
frame.origin.y = 10.0f;
frame.size.width = 100.0f;
frame.size.height = 150.f;

CGrect是C结构体,定义为:

struct CGRect {
   CGPoint origin;
   CGSize size;
};
typedef struct CGRect CGRect;

(分割线)

这里穿插一下Objective-C中关于枚举的定义和用法。例如:

typedef enum {
   kCircle,
   kRectange,
   kOblateSpheroid
}ShapeType;

通常对于枚举的简单声明为:

enum tagname{ ... };

通过上面这种方式声明了一个名为tagname的枚举类型。在C或者Objective-C中如果要使用这一枚举,必须在前面记上关键字enum。

enum tagname x;//declare x of type 'enum tagname'
tagname x;//Error in C/Objective-c,OK in C++;

为了避免总是使用enum,于是出现了下面这种定义:

enum tagname { ... }
typedef enum tagname tagname;//declare 'tagname' as a typedef for 'enum tagname'

进而更简单的方法为:

typedef enum tagname { ... } tagname;//declare both 'enum tagname' and 'tagname'

最终我们可以这样声明:

typedef enum { ... } tagname;

关于最上方的定义官方提供另一种便捷的宏NS_ENUM;

typedef NS_ENUM(NSUInteger, ShapeType) {
   kCircle;
   kRectagle;
   kOblateSpheroid;
}

要点:

  • Objective-C采用动态绑定的消息结构,在运行时才会检查对象类型。接受一条消息后,究竟应该执行何种代码,由运行期环境而非编译器决定。

你可能感兴趣的:(Effective Objective-C(一))