《Effective Objective-C 2.0》1.熟悉 Objective-C

第一章 熟悉 Objective-C

第1条:了解 Objective-C 语言的起源

  • Objective-C 语言在 C 语言基础上添加了面向对象特性。
  • Objective-C 语言由 Smalltalk 演化而来,后者是消息型语言的鼻祖。
  • Objective-C 与 C++、Java 等面向对象语言类似,但是在语法上使用 消息结构 (messaging structure),而非 函数调用 (function calling)。
// Message (Objective-C)
Object *obj = [Object new];
[obj performWith:parameter1 and:parameter2];

// Function calling (C++)
Object *obj = new Object;
obj ->perform(parameter1, parameter2)
  • 区别:使用消息结构的语言,其运行时所应执行的代码由运行环境来决定;而使用函数调用的语言,则由编译器决定。
  • Objective-C 的重要工作都由运行期组件(runtime component)而非编译器来完成。运行期组件本质上就是一种与开发者所编代码相链接的动态库(dynamic library)。
  • Objective-C 语言中的指针是用来指示对象的。而对象所占内存总是分配在堆空间(heap space)。
  • 分配在堆中的内存必须直接管理,而分配在栈上用于保存变量的内存则会在其栈帧弹出时自动清理。
  • Objective-C 使用引用计数来管理堆内存。
  • 定义中不含 * 的变量使用的是栈空间(stack space)。

要点

  • Objective-C 为 C 语言添加了面向对象特性,是其超集。Objective-C 使用动态绑定的消息结构,也就是说,在运行时才会检查对象类型。接收一条消息之后,究竟应执行何种代码,由运行期环境而非编译器来决定。
  • 理解C语言的核心概念有助于写好 Objective-C 程序。尤其要掌握内存模型与指针。

第2条:在类的头文件中尽量少引入其他头文件。

前向声明:在编译一个使用了 EOCPerson 类的文件时,不需要知道 EOCEmployer 类的全部细节,只需要知道有一个类名叫 EOCEmployer 就好。

后向引用:EOCPerson 类的实现文件则需要引入 EOCEmployer 类的头文件,因为若要使用后者,则必须知道其所有接口细节。

将引入头文件的时机尽量延后,只在确有需要时才引入,这样就可以减少类的使用者所需引入的头文件数量

//  EOCPerson.h
#import <Foundation/Foundation.h>

@class EOCEmployer;

@interface EOCPerson : NSObject
@property (nonatomic, copy) NSString *firstName;
@property (nonatomic, copy) NSString *lastName;
@property (nonatomic, strong) EOCEmployer *employer;
@end

//  EOCPerson.m
#import "EOCPerson.h"
#import "EOCEmployer.h"

@implementation EOCPerson

@end

必须在头文件中引入其他头文件:

  • 子类继承父类,则必须引入定义父类的头文件。
  • 声明你写的类遵从某个协议(protocol),那么该协议必须有完整定义。

要点

  • 除非确有必要,否则不要引入头文件。一般来说,应在某个类的头文件中使用向前声明来提及别的类,并在实现文件中引入那些类的头文件。这样做可以尽量降低类之间的耦合(coupling)。
  • 有时无法使用向前声明,比如要声明某个类遵循一项协议。这种情况下,尽量把 “该类遵循某协议” 的这条声明移至 “class-continuation 分类”中。如果不行的话,就把协议单独放在一个头文件中,然后将其引入。

第3条:多用字面量语法,少用与之等价的方法

使用字面量语法(literal syntax)可以缩减源代码长度,使其更为易读。

NSString

// 字符串字面量
NSString *string = @"Effective Objective-C 2.0";

NSNumber

// 使用字面量语法
NSNumber *intNumber    = @1;
NSNumber *floatNumber  = @2.5f;
NSNumber *doubleNumber = @3.14159;
NSNumber *boolNumber   = @YES;
NSNumber *charNumber   = @'a';

int x = 5;
float y = 6.32f;
NSNumber *expressionNumber = @(x * y);

NSArray

// 标准语法
NSArray *animals = [NSArray arrayWithObjects:@"cat",
                                             @"dog",
                                             @"mouse",
                                             @"badger",nil];
NSString *dog = [animals objectAtIndex:1];

// 使用字面量语法
NSArray *animals = @[@"cat",@"dog",@"mouse",@"badger"];
NSString *dog = animals[1];

NSDictionary

// 标准语法
// 顺序:<对象>,<键>,<对象>,<键>,nil
NSDictionary *personData = [NSDictionary dictionaryWithObjectsAndKeys:
                                    @"Matt",@"firstName",
                                    @"Galloway",@"lastName",
                                    [NSNumber numberWithInteger:28],@"age",
                                    nil];
NSString *lastName = [personData objectForKey:@"lastName"];

// 字面量语法
// 顺序:,<键>,<对象>,<键>,<对象>,
NSDictionary *personData = @{@"firstName":@"Matt",
                             @"lastName" :@"Galloway",
                             @"age"      :@"@28"};
NSString *lastName = personData[@"lastName"];

NSMutableArray & NSMutableDictionary

// 修改可变数组与字典内容
[mutableArray replaceObjectAtIndex:1 withObject:@"dog"];
[mutableDictionary setObject:@"Galloway" forKey:@"lastName"];

//下标操作
mutableArray[1] = @"dog";
mutableDictionary[@"lastName"] = @"Galloway";

💡NSURL

你知道 NUSRL 也有字面量语法吗?@@ 是创建 NSURL 的字面量的绝佳方法:

// 标准语法
NSURL *url = [NSURL URLWithString:@"http://example.com"];

// 字面量语法
NSURL *url = @@"http://example.com";

要点

  • 应该使用字面量语法来创建字符串、数值、数组、字典。与创建此类对象的常规方法相比,这么做更加简明扼要。
  • 应该通过取下标操作来访问数组下标或字典中的键所对应的元素。
  • 用字面量语法创建数组或字典时,若值中有 nil,则会抛出异常。因此,务必确保值里不含 nil。

第4条:多用类型常量,少用 #define 预处理指令

  • #define 预处理指令只是简单的替换,定义出来的常量没有类型信息

推荐方法:

// In the implementation file
static const NSTimeInterval KAnimationDuration = 0.3;

常用的命名法:若常量局限于实现文件之内,则在前面加字母 K;若常量在类之外可见,则通常以类名为前缀。

const:如果修改由 const 修饰符所声明的变量,编译器就会报错。

static:该变量仅在定义此变量的编译单元中可见。

对外公开某个常量:

示例一:

// In the header file
extern NSString *const EOCStringConst;

// In the implementation file
NSString *const EOCStringConst = @"VALUE";

示例二:

// EOCAnimatedView.h
extern const NSTimeInterval EOCAnimatedViewAnimationDruation;

// EOCAnimatedView.m
const NSTimeInterval EOCAnimatedViewAnimationDruation = 0.3;

💡推荐使用 UIKIT_EXTERN 关键字代替 extern

要点

  • 不要用预处理指令定义常量。这样定义出来的常量不含类型信息,编译器只是会在编译前据此执行查找与替换操作。即使有人重新定义了常量值,编译器也不会产生警告信息,这将导致应用程序中的常量值不一致。
  • 在实现文件中使用 static const 来定义只在编译单元内可见的常量。由于此类常量不在全局符号表中,所以无需为其名称加前缀。
  • 在头文件中使用 extern 来声明全局常量,并在相关实现文件中定义其值。这种常量要出现在全局符号表中,所以其名称要加以区隔,通常用与之相关的类名做前缀。

第5条:用枚举表示状态、选项、状态码

在用一系列常量来表示错误状态码或可组合的选项时,极宜使用枚举为其命名。

Foundation 框架中的辅助宏定义枚举类型:

方法一,普通枚举类型:

nsenum - Enumerated Type Declaration(NS_ENUM)

typedef NS_ENUM(NSUInteger, EOCConnectionState) {
    EOCConnectionStateDisconnected,
    EOCConnectionStateConnecting,
    EOCConnectionStateConnected,
};

方法二,需要用 按位或操作 来组合的枚举定义:

nsoptions - Enumerated Type Declaration(NS_OPTIONS)

typedef NS_OPTIONS(NSUInteger, EOCPermittedDirection) {
    EOCPermittedDirectionUP    = 1 << 0,
    EOCPermittedDirectionDown  = 1 << 1,
    EOCPermittedDirectionLeft  = 1 << 2,
    EOCPermittedDirectionRight = 1 << 3,
};

方法二用在定义可以组合的选项:

示例:

typedef enum UIViewAutoresizing : NSUInteger {
    UIViewAutoresizingNone                 = 0,
    UIViewAutoresizingFlexibleLeftMargin   = 1 << 0,
    UIViewAutoresizingFlexibleWidth        = 1 << 1,
    UIViewAutoresizingFlexibleRightMargin  = 1 << 2,
    UIViewAutoresizingFlexibleTopMargin    = 1 << 3,
    UIViewAutoresizingFlexibleHeight       = 1 << 4,
    UIViewAutoresizingFlexibleBottomMargin = 1 << 5
} UIViewAutoresizing;

// 通过「按位或操作符」组合多个选项
UIViewAutoresizing resizing = UIViewAutoresizingFlexibleWidth | UIViewAutoresizingFlexibleHeight;
// 通过「按位与操作符」判断是否已启用某个选项
if (resizing & UIViewAutoresizingFlexibleWidth) {
    // UIViewAutoresizingFlexibleWidth is set
}

要点

  • 应该用枚举来表示状态机的状态、传递给方法的选项以及状态码等值,给这些值起个易懂的名字。
  • 如果把传递给某个方法的选项表示为枚举型,而多个选项又可同时使用,那么就将各选项值定义为2的幂,以便通过按位或者操作将其组合起来。
  • NS_ENUMNS_OPTIONS 宏来定义枚举类型,并指明其底层数据类型。这样做可以确保枚举是用开发者所选的底层数据类型实现出来的,而不会采用编译器所选的类型。
  • 在处理枚举类型的 switch 语句中不要实现 default 分支。这样的话,加入新枚举之后,编译器就会提示开发者:switch 语句并未处理所有的枚举。
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,098评论 5 476
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,213评论 2 380
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 149,960评论 0 336
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,519评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,512评论 5 364
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,533评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,914评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,574评论 0 256
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,804评论 1 296
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,563评论 2 319
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,644评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,350评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,933评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,908评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,146评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,847评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,361评论 2 342

推荐阅读更多精彩内容

  • importUIKit classViewController:UITabBarController{ enumD...
    明哥_Young阅读 3,771评论 1 10
  • 1. Java基础部分 基础部分的顺序:基本语法,类相关的语法,内部类的语法,继承相关的语法,异常的语法,线程的语...
    子非鱼_t_阅读 31,560评论 18 399
  • 这是真的。 在那个酷寒的冬夜,它披着柔滑却稀少的白色毛发,卷缩在车胎旁,它望着零零散散的人匆忙的走...
    羽落y阅读 201评论 0 0
  • 第七十八章 不寻常的晚宴 慕瑶汐也没打算隐瞒什么,于是薄唇轻启缓缓道:’之前我拜托过翟毅帮忙寻找白家老爷子中毒...
    殇祁祁阅读 353评论 0 1
  • 旭旭是新来的三岁男孩,耳闻他不太会表达自己,不会与他人相处,喜欢打滚、大叫。但前几天我都没有直接接触,只偶尔看到他...
    雾嘉花阅读 561评论 0 1