Object-C总结摘录

11年前

1、  Object-C语言是由Brad J.Cox于20世纪80年代早期设计的,它是以SmallTalk-80语言为基础的。但是,Object-C是C语言的扩展,它是C的超集。

2、  在终端使用gcc编译一个Object-C源文件main.m,命令如下:

gcc –framework Fundation main.m –o man

3、  Object-C入口函数为main。

4、  4种基本数据类型:int float double char ,还有其它3种特殊的数据类型:_Bool、_Complex、_Imaginary,分别用于处理布尔值、复数及抽象数字。

使用布尔数据类型时,经常使用BOOL,这种“数据类型”其实并不是真正的数据类型,而只是char数据类型的别名,通过typedef实现的。

5、  单行注释用//,代码块注释用 /**/,但是/*注释不能嵌套使用。

6、  变量名称必须以字母或_开头。

7、  Object-C中大小写字母是由区别的。即:int idx,Idx 表示两个不同的变量。

8、  #import <Foundation/Foundation.h>表示引入系统头文件;

#import “Cat.h”,表示引入本地头文件。

9、  -(void)print,表示是一个实例方法;+(void)print,表示是一个类方法。

10、              使用关键字@interface声明一个类,比如Cat.h:

@interface Cat:NSObject //表示Cat类的父类是NSObject

@property int age;//1

@end //表示声明结束

从Object-C 2.0开始,上面代码1可自动生成设置age的函数及获取age的函数(即:setter、getter方法),其实现是语言层面的,是由Object-C 2.0的编译器负责生成的。

但是,Cat.m文件中要使用关键字@synthesize指令,

#import “Cat.h”

@implementation Cat // implementation,是实现方法的关键字

@synthesize age;

 

@end

11、              Cat *cat = [Cat new],其中的new方法是将alloc与init操作合并在一起了,应该尽量不用这种方式。

12、              多参数函数声明:-(void)setWidth:(int) width  height:(int) height;

第一个height为setWidth函数第二个参数的名称,第二个height为形参名称。

参数名称是可选的,为了程序的可读性更好,应该使用带参数名称的函数声明方式。

13、              类的每个实例都拥有实例变量,即使这些实例变量时继承来的。

14、              分类(Category)允许以模块方式向现类定义添加新方法(不可以添加实例变量);当要对没有源代码访问权限的类添加新定义时,可以使用分类技术。

当然也可以使用继承,怎么选择使用分类还是继承呢?

我的理解:继承会增加对 象间的复杂度,一旦继承某个类,会继承该类的所有实例变量和实例方法,但是并不一定都需要,这时继承这一堆东西就是浪费。1)如果仅仅关注要扩展的该类缺 失的功能,并不太关注该类其他的数据和功能,那么应该使用分类(Category),否则可选择使用继承;2)如果想为该类扩展数据,那只能选择继承了, 因为分类无法扩展实例变量,而只能扩展方法。

15、              @class指令,比如:@class Cat; 这时编译器不再需要处理整个Cat.h头文件,会适当提高效率,但是如果当前类需要引用Cat类中的方法,仅用@class指令时不够的,因为编译器需要 更多信息,比如:该方法中有多少参数,参数是什么类型,方法的返回类型等,而@class仅仅是提供了一个已知的类名称,此时只能用#import “Cat.h”。

16、              抽象类:创建该类的目的是为了使用其子类,而不是直接使用该类。

Foundation框架中的NSNumber就是为了将数字作为对象处理而创建的一个抽象类。

整数与浮点数通常有不同的内存需求,每种数字类型都有单独的NSNumber子类。向NSNumber类发送消息来创建新的整数对象时,使用合适的子类为整数对象分配必须的空间,并正确地设定其值。

但是,这些子类是私有的,无法直接访问这些子类,只能通过抽象的超类间接访问。

17、              多态:不同类的对象可以定义共享相同名称。如:id数据类型可以用来存储属于任何类的对象。

18、              动态类型:使程序直到执行时才确定对象所属的类。

19、              动态绑定:使程序直到执行时才确定要对对象调用的方法。

20、              如果一个类包含多个初始化方法,那么其中一个应该是指定的初始化方法,并且其它所有初始化方法都使用这个方法。

21、              分类:

1)  分类可以访问原始类的实例变量,但是不能添加自身的任何变量。

2)  虽然分类可以重写该类中的方法,但是通常不这么做。如果确实需要重写方法,应该选择继承而不是分类。

3)  不必实现分类中的所有方法。

4)  通过使用分类添加方法扩展类不仅会影响这个类,同时也会影响它的所有子类。

5)  对象/分类命名对必须是唯一的。

22、              协议:是多个类共享的一个方法列表(类比java中的接口)。

23、              C-string由char字符组成;NSString对象由unichar字符组成,unichar字符时符合Unicode标准的多字节字符。

24、              NSObject中有个description方法(类比java中的toString方法)。

25、              NSString:创建内容不可更改的对象;NSMutableString:创建内容可以更改的对象;

类似的还有:

NSArray与NSMutableArray,NSDictionary与NSMutableDictionary;NSSet与NSMutableSet(一组单值对象集合)

26、              字典的键必须是单值的,尽管通常是字符串,但可以使任何对象类型。和键关联的值可以任何对象类型,但不可以为nil。

27、              如果obj1与obj2是同一对象,说明它们占用相同的内存位置。

28、              NSFileManager功能:

1)  创建一个新文件

2)  将数据写入文件/从现有文件中读取数据

3)  重命名文件

4)  删除、复制文件

5)  读取文件的大小等其他属性

6)  测试文件是否存在及两个文件内容是否相同等

以上操作也可针对目录。

29、              NSFileHandle:可以打开文件并对其执行多次读写操作,

1)  打开一个文件,执行读、写或更新操作

2)  在文件中查找指定位置

3)  从文件中读取特定数目的字节,或将特定数目的字节写入文件中

4)  也可以用于各种设备或套接字

30、              NSData:一个临时存储区,作为缓冲区,将文件的内容读入缓冲区,或将缓冲区的内容写入到一个文件。

31、              NSFIleManager枚举目录,方法enumerratorAtPath自动递归枚举子目录;方法directoryContentsAtPath方法不递归枚举。

32、              读写文件设置的缓冲区buffer数组一般都是2的乘方,这是因为底层操作系统通常以块为单位执行I/O操作的,而块的大小一般为2的乘方个字节。

33、              引用计数:跟踪对象的引用次数。

1)  将对象添加到任何类型的集合都会使该对象的引用次数增加。

2)  把对象赋值给另一个变量时,引用次数不会自动增加,所以必须程序员自己增加对象的引用次数。

3)  只要对象的引用计数大于0,系统就不会释放对象占有的内容。

4)  内存中常量字符串的空间分配与其他对象不同,它们没有引用计数机制,因此永远不能释放这些对象

34、              自动释放池可以自动释放添加到该池中的对象所使用的内存。向对象发送一条autorelease消息时,就将该对象添加到这个池中。

35、              自动释放池会嵌套使用。

36、              将对象添加到自动释放池中并没有释放它或使它无效,这仅仅是为以后释放它做了个标记。当池被释放时,如果对象的引用值恰好变为0,则释放对象的资源(所以,如果池被释放时,对象的引用值大于0,那么就不会释放对象的资源)。

37、              内存管理规则:

1)  不再使用创建或保持的对象时,就释放它们。

2)  发送一条release消息,不一定会销毁对象;只有当一个对象的计数变为0时,才能销毁这个对象。

3)  每次释放池时,系统通过向池中的每一个对象发送一条release消息来实现这个功能。同时,系统会向池中每个引用计数为0的对象发送一条dealloc消息来销毁这个对象。

4)  如果你的方法中不再使用一个对象,但需要返回它,那么向其发送一条autorelease消息,将它标记为以后释放;消息autorelease并不会影响对象的引用计数。因此,它允许消息的发送者使用这个对象,然而仍然在以后自动释放池时,释放这些对象。

5)  无论对象是否添加到自动释放池中,应用程序终止时,都会释放程序中对象占用的所有内存。

6)  如果想要保证在自动释放池被释放时对象仍然存在,则应该显示地保持它。如果对象的引用计数大于收到的autorelease消息总数目,则在池释放时会保留下来。

7)  如果使用alloc或copy方法(allocWithZone、copyWithZone、mutableCopy方法)直接创建对象,则由你负责释放对象。每次retain对象时,应该release或者autorelease它。

8)  除了上一条提到的方法,不用费心去其他方法返回的对象。这些对象应该被它们的方法自动释放,这就是首先在程序中创建自动释放池的原因。 stringWithString之类的方法自动向新创建的字符串对象发送一条autorelease消息,把它们添加到自动释放池中。如果事先没有创建 池,则会收到一条出错消息,意思是在没有池的情况下尝试自动释放对象。

38、              深浅复制相对对象而言,浅复制仅复制对象的指针,副本与原对象指针都指向相同的内存。深复制,不仅复制对象的指针,而且将原对象指针对应的内存内容完全复制一份,复制后的副本与原对象占用不同的内存地址,但是内存内容完全相同。

39、              产生一个对象的可变副本与一个对象本身是否可变无关。所以,可以创建可变对象的不可变副本;也可以创建不可变对象的可变副本。

40、              在创建数组的副本时,数组中的每个元素的保持计数将通过复制操作自动增1。因此,如果创建数组的副本并随即释放原始数组,那么副本仍然包含有效的元素。

41、              如果编写一个类的copyWIthZone:方法,而该类的超类也实现了<NSCopying>协议,那么应该先调用超类的copy方法以复 制继承来的实例变量,然后加入自己的代码以复制想要添加到该类的任何附加的实例变量。

42、              编写一个类的赋值方法和取值方法时,一定要考虑实例变量中存储的内容,要检索的内容以及是否要保护这些内容。

43、              归档:用某种格式保存一个对象或多个对象,以便以后还原这些对象的过程。

44、              可以使用归档实现对象的复制,因为在归档和解档过程中产生对象的新副本。

45、              框架是一组从逻辑上组合在一起的类和例程。