OC的起源
OC是一种消息型语言,使用的是“消息结构”而非“函数调用”,由smalltalk演化而来。使用消息结构的语言运行时执行的代码由运行环境来决定,而使用函数调用的语言由编译器决定。
什么是引用计数
OC将堆内存管理抽象出来了。不需要使用malloc或者free来分配或释放对象所占的内存。OC运行期环境把这部分工作抽象成一套内存管理架构,名为“引用计数”。
- OC为C语言添加了面向对象特性,是其超集。OC使用动态绑定的消息结构,也就是说在运行的时候才会检查对象类型。接收一条消息后,究竟应执行哪种代码,由运行期环境而非编译器来决定。
向前声明某类
当编译一个A类的文件时,想要引入B类,但是不需要知道B类的全部细节,只需要知道有一个类叫B类,可以使用:
@class B;
但是如果要在A类的实现文件中使用B,则需要使用import声明,因为要使用它就要知道B类的所有细节。
这样做延后了引入头文件的时机,只有在确有需要时才引入,这样就可以减少类的使用者所需引入的头文件数量,减少编译时间。
向前声明同样可以解决两个类互相引用的问题。在两个类中互相引用对方的头文件,就会造成“循环引用”。当解析一个头文件时,会发现它引用了另一个头文件,而那个头文件又回头引用了第一个头文件。使用#import而非#include虽然不会导致死循环,但是会使得两个类中有一个无法被正常编译。
如果你写的类继承自某个超类,则必须引入定义那个超类的头文件,同理如果要声明你写的类遵从某个协议,那么该协议必须有完整定义,且不能向前声明。因为向前声明只是声明某个协议的存在,而此时编译器却需要知道该协议的具体定义。
- 除非确实有必要,否则不要引入头文件。一般来说应在某个类的头文件中使用向前声明来提及别的类,并在实现文件中引入那些类的头文件。这样做可以降低类之间的耦合。
有时无法使用向前声明,比如要声明某个类遵循一项协议。这种情况下尽量把“该类遵循某项协议”的这条声明放在分类中。如果不行的话就把协议单独放在一个头文件中然后引入。
多用字面量语法
使用字面量语法可以缩减代码长度,使其更为易读。
NSString:
NSString *str = @“...”;
NSNumber:
NSNumber *num = @1;
NSArray:
NSArray *arr = @[@“1”, @“2”, @“3”];
NSString *two = arr[1];
NSDIctionary:
NSDictionary *dict = @{@“1”: @“one”, @“2”: @“two”, @“3”: @“three”};
NSString *two = dict[@“2”];
NSMutableDictionary和NSMutableArray:
//字面量语法修改可变数组或者字典中的元素时可这样修改
mutableArray[1] = @“2”;
mutableDictionary[@“3”] = @“4”;
在数组中使用字面量语法时,倘若我们放入数组的元素是变量,其中一个变量为nil,那么编译器会报出异常,方便我们及时发现问题。但是如果使用的不是字面量语法,比如NSArray *arr = [NSArray arrayWithObjects: obj1, obj2, obj3, nil]; 在这个数组中如果obj2为nil,那么arr的值就会只有obj1,它不会报出异常,这是因为arrayWithObjects方法处理参数时发现nil就停止了,这也导致我们无法及时发现问题。字典同理。
因此,使用字面量语法也有助于我们快速发现问题。
字面量语法的局限性
字面量语法有个限制就是:除了字符串以外,所创建的对象必须属于Foundation框架才行。意思是自定义的这些类的子类是不能用字面量语法的。
使用字面量语法创建出来的字符串、数组、字典对象都是不可变的。若想要可变版本,需要:
NSMutableArray *mutableArray = [@[@1, @2, @3, @4]mutableCopy];
- 应该使用字面量语法来创建字符串、数组、数值、字典。与常规方法相比。字面量语法更加简明扼要。
应该通过取下标操作来访问数组下标或者字典中的键对应的元素。
用字面量语法创建数组或字典时,若值中有nil,则会抛出异常。
多使用常量,少使用#define
定义常量应用static const,这样方式定义的常量包含类型信息,可以清楚的了解常量的含义。
定义的常量不应该放在头文件里。如果不打算公开某个常量,则应该将其定义在实现文件里。
变量一定要同时用static和const声明,如果试图修改由const修饰符所声明的变量,那么编译器就会报错。
有时候需要对外公开某个常量,比如使用通知传值的时候要用一个对象来派发通知,然后让其他接收通知的对象来注册通知完成传值。在派发通知的时候需要使用一个字符串来表明这个通知的名称,这个名称就可以声明为一个外界可见的常值变量。这样注册者就无需知道实际字符串的值,只需要以常值变量来注册自己想要接收的通知就行。这一类常量应该放在“全局符号表”中,以便可以在定义该常量的编译单元之外使用。
定义在全局符号表中:
extern NSString *const stringConst;
NSString *const stringConst = @“value”;
//在上面的代码中,stringConst是一个指针常量,指向NSString对象。
- 不要用预处理指令定义常量。这样定义出来的常量不含类型信息,编译器只是会在编译前据此执行查找与替换操作。即使有人重新定义了常量值,编译器也不会产生警告信息,这将导致应用程序中的常量值不一致。
在实现文件中使用static const来定义“只在编译单元内可见的常量”,由于此类常量不在全局符号表中,所以无须为其命名加前缀。
在头文件中使用extern来声明全局变量,并在相关实现文件中定义其值。这种常量要出现在全局符号表中,所以其命名应该加一区隔,通常用与之相关的类名做前缀。
用枚举类型
在以一系列常量来表示错误状态码或可组合选项时,宜用枚举为其命名。
定义枚举的语法:
enum connectionState state = disconnected;
enum connectionState {
disconnected,
connected,
connecting,
};
typedef enum connectionState connectionState;
connectionState state = disconnected;
可以指明用何种“底层数据类型”来保存枚举值的变量。因此这样做就可以向前声明枚举变量。若不指明底层数据类型就不可以向前声明,因为编译器不清楚底层数据类型的大小,不知道要给它分配多少空间。
指定底层数据类型:
enum connectionState: NSInteger{...};
在向前声明时指定底层数据类型:
enum connectionState: NSInteger;
不使用编译器分配的序号,使用自定义指定的值:文章来源:https://www.toymoban.com/news/detail-795795.html
enum connetcionState {
//将disconnected的值设为1,而不是编译器指定的0,这样接下来几个枚举的值都会是上一个的值+1
disconnected = 1,
connected,
connecting,
};
定义选项时可以通过枚举组合选项,各选项之间通过“按位或操作符”来组合。文章来源地址https://www.toymoban.com/news/detail-795795.html
- 应该用枚举来表示状态机的状态、传递给方法的选项以及状态码等值,给这些值起个易懂的名字。
如果把传递给某个方法的选项表示为枚举类型,而多个选项又可以同时使用,那么就将各选项值定义为2的幂,以便通过按位或操作将其组合起来。
用NS_ENUM与NS_OPTIONS宏来定义枚举类型,并指明其底层数据类型。这样做可以确保枚举是用开发者所选的底层数据类型实现出来的,而不会采用编译器所选的类型。
在处理枚举类型的switch语句中不要实现default分支,这样的话,加入新枚举之后,编译器就会提示开发者switch语句并未处理所有枚举。
到了这里,关于Effective Objective-C学习第一周的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!