今天我们即将结束了对Objective-C的学习,虽然不想承认,但还是觉得自己对于Objective-C来说还是有很多的知识没有学会,于是就想着在今天对OC这门语言做一个全面的知识点的总结,以方便自己以后能够随时的翻看。如下便是我对Objective-C语言的相关知识点及相关示例代码的总结:(如有不全,请见谅)
Objective-C是一种通用、高级、面向对象的编程语言。它扩展了标准的ANSI C编程语言,将Smalltalk式的消息传递机制加入到ANSI C中。它是苹果的OS X和iOS操作系统,及其相关API、Cocoa和Cocoa Touch的主要编程语言。
Objective-C最初源于NeXTSTEP操作系统,之后在OS X和iOS继承下来。目前主要支持的编译器有GCC和LLVM(采用Clang作为前端),苹果公司在Xcode 4.0之后均采用LLVM作为默认的编译器。最新的Objective-C特性也都率先在Clang上实现。
Objective-C的发展背景:
1980年代初,Stepstone公司的Brad Cox与Tom Love发明了Objective-C。1981年,当他们两人还在ITT公司技术中心任职时,接触到了一个名为SmallTalk-80的语言,Cox当时正巧对软件设计中的重用问题非常感兴趣,他马上就意识到了SmallTalk-80这个语言在软件建构中无法衡量的价值。但与此同时Cox与Love也明白,在目前的ITT公司中,引入技术最重要的关键是能与C语言兼容。
于是Cox开始撰写一个C语言的前处理器,使C语言具备了些许Smalltalk的本领。Cox很快地实现了一个能用的C语言扩展,此即为Objective-C语言的前身。到了1983年,Cox与Love合伙成立了ProductivityProducts International(PPI)公司,将Objective-C及其相关库商品化贩售,之后公司改名为StepStone。1986年,Cox出版了一本重要著作《Object-Oriented Programming, An Evolutionary Approach》,详述了Objective-C的种种设计理念。
1988年,乔布斯(Steve Jobs)离开苹果公司后成立了NeXT Computer公司,NeXT公司买下了Objective-C语言的授权,并扩展了著名的开源编译器GCC使之能支持Objective-C。并用Objective-C开发了AppKit与Foundation Kit等等库,作为NeXTSTEP的的用户接口与开发环境的基础。后来NeXT工作站在市场上失败后,NeXT上的软件工具却在业界中被广泛赞扬。这促使NeXT公司放弃硬件业务,转型为销售NeXTStep(以及OpenStep)平台为主的软件公司。
1992年,自由软件基金会的GNU开发环境增加了对Objective-C的支持。1994年,NeXT Computer公司和Sun Microsystem联合发布了一个针对NEXTSTEP系统的标准典范,名为OPENSTEP。OPENSTEP在自由软件基金会的实现名称为GNUstep。1996年12月20日,苹果公司宣布收购NeXT Software公司,NEXTSTEP/OPENSTEP环境成为苹果操作系统下一个主要发行版本OS X的基础。这个开发环境的该版本被苹果公司称为Cocoa。
Objective-C语法:
Objective-C是C语言的严格母集合,意思是任何原始的C语言程序不经修改就可以直接通过Objective-C编译器,在Objective-C中使用C语言代码也是完全合法的。Objective-C形容自己是盖在C语言上的薄薄一层,因为Objective-C的原意就是在C语言主体上加入面向对象的特性。Objective-C的面向对象语法源于Smalltalk信息传递风格。所有其他非面向对象的语法,包括变量类型,预处理器(preprocessing),流程控制,函数声明与调用皆与C语言完全一致。
1、Hello World
这里示范了一个基础的Hello World程序。基于Xcode 4.3.1 xcode:
#import
int main(int argc,char *argv[]){
@autoreleasepool{
NSLog(@"HelloWorld!");
}
return 0;
}
2、信息传递
Objective-C最大的特色是承自Smalltalk的信息传递模型(message passing),此机制与今日C++式之主流风格差异甚大。Objective-C里,与其说对象互相调用方法,不如说对象之间互相传递信息更为精确。此二种风格的主要差异在于调用方法/传递模型这个动作。C++里类型与方法的关系严格清楚,一个方法必定属于一个类型,而且在编译时(compile time)就已经紧密绑定,不可能调用一个不存在类型里的方法。但在Objective-C,类型与信息的关系比较松散,调用方法视为对对象发送信息,所有方法都被视为对信息的回应。所有信息处理直到运行时(runtime)才会动态决定,并交由类型自行决定如何处理收到的信息。也就是说,一个类型不保证一定会回应收到的信息,如果类型收到了一个无法处理的信息,程序只会抛出异常,不会出错或崩溃。
C++里,送一个信息给对象(或者说调用一个方法)的语法如下:
obj->method(argument);
Objective-C则写成:
[objmethod: argument];
此二者并不仅仅是语法上的差异,还有基本行为上的不同。
这里以一个汽车类(car class)的简单例子来解释Objective-C的信息传递特性:
[carfly];
典型的C++意义解读是“调用car类型的fly方法”。若car类型里头没有定义fly方法,那编译肯定不会通过。但是Objective-C里,我们应当解读为“发提交一个fly的信息给car对象”,fly是信息,而car是信息的接收者。car收到信息后会决定如何回应这个信息,若car类型内定义有fly方法就运行方法内之代码,若car内不存在fly方法,则程序依旧可以通过编译,运行期则抛出异常。
此二种风格各有优劣。C++强制要求所有的方法都必须有对应的动作,且编译期绑定使得函数调用非常快速。缺点是仅能借由virtual关键字提供有限的动态绑定能力。Objective-C天生即具备鸭子类型之动态绑定能力,因为运行期才处理信息,允许传送未知信息给对象。可以送信息给整个对象集合而不需要一一检查每个对象的型态,也具备消息转送机制。同时空对象nil接受信息后默认为不做事,所以送信息给nil也不用担心程序崩溃。
Objective-C的方法调用因为运行期才动态解析信息,一开始信息比C++ virtual成员函数调用速度慢上三倍。但经由IMP高速缓存改善,目前已经比C++的virtual function快上50%。
3、类的定义与实现
Objective-C中强制要求将类的定义(interface)与实现(implementation)分为两个部分。
类的定义文件遵循C语言之惯例以.h为后缀,实现文件以.m为后缀。
4、Interface
定义部分,清楚定义了类的名称、数据成员和方法。 以关键字@interface作为开始,@end作为结束。
@interface MyObject: NSObject {
int memberVar1;
id memberVar2;
}
+(return_type) class_method;// 类方法
-(return_type) instance_method1;//实例方法
-(return_type) instance_method2:(int) p1;
-(return_type) instance_method3:(int) p1 andPar: (int) p2;
@end
方法前面的 +/- 号代表函数的类型:加号(+)代表类方法(class method),不需要实例就可以调用,与C++ 的静态函数(static member function)相似。减号(-)即是一般的实例方法(instance method)。
Objective-C定义一个新的方法时,名称内的冒号(:)代表参数传递,不同于C语言以数学函数的括号来传递参数。Objective-C方法使得参数可以夹杂于名称中间,不必全部附缀于方法名称的尾端,可以提高程序可读性。以设置颜色RGB值的方法为例:
-(void) setColorToRed:(float)red Green: (float)green Blue:(float)blue;/* 宣告方法*/
[myColorsetColorToRed:1.0Green: 0.8Blue: 0.2];/* 呼叫方法*/
这个方法的签名是setColorToRed:Green:Blue:。每个冒号后面都带着一个float类型的参数,分别代表红,绿,蓝三色。
5、Implementation
实现区段则包含了公开方法的实现,以及定义私有(private)变量及方法。以关键字@implementation作为区段起头,@end结尾。
@implementation MyObject{
int memberVar3; //私有
}
+(return_type) class_method{
.... //method implementation
}
-(return_type) instance_method1{
....
}
-(return_type) instance_method2:(int) p1 {
....
}
-(return_type) instance_method3:(int) p1 andPar: (int) p2{
....
}
@end
值得一提的是不只Interface区段可定义实体变量,Implementation区段也可以定义实体变量,两者的差别在于访问权限的不同,Interface区段内的实体变量默认权限为protected,声明于implementation区段的实体变量则默认为private,故在Implementation区段定义私有成员更符合面向对象之封装原则,因为如此类型之私有信息就不需曝露于公开interface(.h文件)中。
6、创建对象
Objective-C创建对象需通过alloc以及init两个消息。alloc的作用是分配内存,init则是初始化对象。 init与alloc都是定义在NSObject里的方法,父对象收到这两个信息并做出正确回应后,新对象才创建完毕。以下为范例:
MyObject * my = [[MyObjectalloc] init];
在Objective-C 2.0里,若创建对象不需要参数,则可直接使用new
MyObject * my = [MyObject new];
仅仅是语法上的精简,效果完全相同。
若要自己定义初始化的过程,可以重写init方法,来添加额外的工作。(用途类似C++ 的构造函数constructor)
-(id) init{
if(self=[super init]){ // 必须调用父类的init
// do something here ...
}
return self;
}
7、协议(Protocol)
协议是一组没有实现的方法列表,任何的类均可采纳协议并具体实现这组方法。
Objective-C在NeXT时期曾经试图引入多重继承的概念,但由于协议的出现而没有实现之。
协议类似于Java与C#语言中的“接口”。在Objective-C中,有两种定义协议的方式:由编译器保证的“正式协议”,以及为特定目的设定的“非正式协议”。
非正式协议为一个可以选择性实现的一系列方法列表。非正式协议虽名为协议,但实际上是挂于NSObject上的未实现分类(Unimplemented Category)的一种称谓,Objetive-C语言机制上并没有非正式协议这种东西,OSX 10.6版本之后由于引入@optional关键字,使得正式协议已具备同样的能力,所以非正式协议已经被废弃不再使用。
正式协议类似于Java中的"接口",它是一系列方法的列表,任何类都可以声明自身实现了某个协议。在Objective-C 2.0之前,一个类必须实现它声明符合的协议中的所有方法,否则编译器会报告错误,表明这个类没有实现它声明符合的协议中的全部方法。Objective-C 2.0版本允许标记协议中某些方法为可选的(Optional),这样编译器就不会强制实现这些可选的方法。
协议经常应用于Cocoa中的委托及事件触发。例如文本框类通常会包括一个委托(delegate)对象,该对象可以实现一个协议,该协议中可能包含一个实现文字输入的自动完成方法。若这个委托对象实现了这个方法,那么文本框类就会在适当的时候触发自动完成事件,并调用这个方法用于自动完成功能。
Objective-C中协议的概念与Java中接口的概念并不完全相同,即一个类可以在不声明它符合某个协议的情况下,实现这个协议所包含的方法,也即实质上符合这个协议,而这种差别对外部代码而言是不可见的。正式协议的声明不提供实现,它只是简单地表明符合该协议的类实现了该协议的方法,保证调用端可以安全调用方法。
语法
协议以关键字@protocol作为区段起始,@end退出,中间为方法列表。
@protocol Locking
-(void)lock;
-(void)unlock;
@end
这是一个协议的例子,多线程编程中经常要确保一份共享资源同时只有一个线程可以使用,会在使用前给该资源挂上锁 ,以上即为一个表明有“锁”的概念的协议,协议中有两个方法,只有名称但尚未实现。
下面的SomeClass宣称他采纳了Locking协议:
@interface SomeClass:SomeSuperClass
@end
一旦SomeClass表明他采纳了Locking协议,SomeClass就有义务实现Locking协议中的两个方法。
@implementation SomeClass
-(void)lock{
// 实现lock方法...
}
-(void)unlock{
// 实现unlock方法...
}
@end
由于SomeClass已经确实遵从了Locking协议,故调用端可以安全的发送lock或unlock信息给SomeClass实体变量,不需担心他没有办法回应信息。
插件是另一个使用抽象定义的例子,可以在不关心插件的实现的情况下定义其希望的行为。
8、动态类型
类似于Smalltalk,Objective-C具备动态类型:即消息可以发送给任何对象实体,无论该对象实体的公开接口中有没有对应的方法。在C++这种静态类型的语言里,不可能对一个(void*)指针调用任何方法,编译器会挡下该调用行为。但在Objective-C中,你可以对id发送任何信息(id很像void*,但是被严格限制只能使用在对象上),编译器仅会发出“该对象可能无法回应信息”的警告,程序同样可以通过编译,而实际发生的事则取决于运行期该对象的真正形态,若该对象的确可以回应消息,则依旧运行对应的方法。
这种特性可以增加语言的灵活性,因为它允许对象“捕捉”消息,再将消息转送到另一个可以正确处理该消息的对象,形同消息“转发”给另一个对象。
一个对象收到信息之后,他有三种处理信息的可能手段,第一是回应该消息并运行方法,若无法回应,则可以转发消息给其他对象,若以上两者均无,就要处理无法回应而抛出的例外。只要进行三者之其一,该消息就算完成任务而被丢弃。若对“nil”(空对象指针)发送消息,该消息通常会被忽略,取决于编译器选项可能会抛出例外。
虽然Objective-C具备动态类型的能力,但编译期的静态类型检查依旧可以应用到变量上。以下三种声明在运行时效力是完全相同的,但是三种声明提供了一个比一个更明显的类型信息,附加的类型信息让编译器在编译时可以检查变量类型,并对类型不符的变量提出警告。
下面三个方法,差异仅在于参数的形态:
-setMyValue:(id) foo;
id形态表示参数“foo”可以是任何类的实例。
-setMyValue:(id
id
-setMyValue:(NSNumber*) foo;
该声明表示“foo”必须是“NSNumber”的实例。
动态类型是一种强大的特性。在缺少泛型的静态类型语言(如Java 5以前的版本)中实现容器类时,程序员需要写一种针对通用类型对象的容器类,然后在通用类型和实际类型中不停的强制类型转换。无论如何,类型转换会破坏静态类型,例如写入一个“整数”而将其读取为“字符串”会产生运行时错误。这样的问题被泛型解决,但容器类需要其内容对象的类型一致,而对于动态类型语言则完全没有这方面的问题。
9、转发
Objective-C允许对一个对象发送消息,不管它是否能够响应之。除了响应或丢弃消息以外,对象也可以将消息转发到可以响应该消息的对象。转发可以用于简化特定的设计模式,例如观测器模式或代理模式。
Objective-C运行时在Object中定义了一对方法:
1. 转发方法:
-(retval_t)forward:(SEL) sel:(arglist_t) args;// with GCC
-(id) forward:(SEL) sel:(marg_list)args;// with NeXT/Apple systems
1. 响应方法:
-(retval_t)performv:(SEL) sel:(arglist_t) args; // with GCC
-(id) performv:(SEL) sel:(marg_list)args;// with NeXT/Apple systems
希望实现转发的对象只需用新的方法覆盖以上方法来定义其转发行为。无需重写响应方法performv::,由于该方法只是单纯的对响应对象发送消息并传递参数。其中,SEL类型是Objective-C中消息的类型。
例子
这里包括了一个演示转发的基本概念的程序示例。
Forwarder.h
#import
@interface Forwarder: Object
{
id recipient; //该对象是我们希望转发到的对象。
}
@property(assign, nonatomic)id recipient;
@end
Forwarder.m
#import "Forwarder.h"
@implementation Forwarder
@synthesize recipient;
-(retval_t)forward:(SEL) sel: (arglist_t) args
{
/*
*检查转发对象是否响应该消息。
*若转发对象不响应该消息,则不会转发,而产生一个错误。
*/
if([recipientrespondsTo:sel])
return[recipientperformv: sel:args];
else
return[selferror:"Recipientdoes not respond"];
}
Recipient.h
#import
// A simple Recipient object.
@interface Recipient: Object
-(id) hello;
@end
Recipient.m
#import "Recipient.h"
@implementation Recipient
-(id) hello
{
printf("Recipient says hello!\n");
return self;
}
@end
main.m
#import "Forwarder.h"
#import "Recipient.h"
int main(void)
{
Forwarder*forwarder =[Forwarder new];
Recipient*recipient =[Recipient new];
forwarder.recipient= recipient; //Set the recipient.
/*
*转发者不响应hello消息!该消息将被转发到转发对象。
*(若转发对象响应该消息)
*/
[forwarder hello];
return 0;
}
10、分类(Category)
在Objective-C的设计中,一个主要的考虑即为大型代码框架的维护。结构化编程的经验显示,改进代码的一种主要方法即为将其分解为更小的片段。Objective-C借用并扩展了Smalltalk实现中的“分类”概念,用以帮助达到分解代码的目的。
一个分类可以将方法的实现分解进一系列分离的文件。程序员可以将一组相关的方法放进一个分类,使程序更具可读性。举例来讲,可以在字符串类中增加一个名为“拼写检查”的分类,并将拼写检查的相关代码放进这个分类中。
进一步的,分类中的方法是在运行时被加入类中的,这一特性允许程序员向现存的类中增加方法,而无需持有原有的代码,或是重新编译原有的类。例如若系统提供的字符串类的实现中不包含拼写检查的功能,可以增加这样的功能而无需更改原有的字符串类的代码。
在运行时,分类中的方法与类原有的方法并无区别,其代码可以访问包括私有类成员变量在内的所有成员变量。
若分类声明了与类中原有方法同名的函数,则分类中的方法会被调用。因此分类不仅可以增加类的方法,也可以代替原有的方法。这个特性可以用于修正原有代码中的错误,更可以从根本上改变程序中原有类的行为。若两个分类中的方法同名,则被调用的方法是不可预测的。
其它语言也尝试了通过不同方法增加这一语言特性。TOM在这方面走的更远,不仅允许增加方法,更允许增加成员变量。也有其它语言使用面向声明的解决方案,其中最值得注意的是Self语言。
C#与Visual Basic.NET语言以扩展函数的与不完全类的方式实现了类似的功能。Ruby与一些动态语言则以"monkey patch"的名字称呼这种技术。
使用分类的例子
这个例子创建了Integer类,其本身只定义了integer属性,然后增加了两个分类Arithmetic与Display以扩展类的功能。虽然分类可以访问类的私有成员,但通常利用属性的访问方法来访问是一种更好的做法,可以使得分类与原有类更加独立。这是分类的一种典型应用—另外的应用是利用分类来替换原有类中的方法,虽然用分类而不是继承来替换方法不被认为是一种好的做法。
Integer.h
#import
@interface Integer: Object
{
@private
int integer;
}
@property(assign, nonatomic)integer;
@end
Integer.m
#import "Integer.h"
@implementation Integer
@synthesize integer;
@end
Arithmetic.h
#import "Integer.h"
@interface Integer(Arithmetic)
-(id) add:(Integer *)addend;
-(id) sub:(Integer *)subtrahend;
@end
Arithmetic.m
#import "Arithmetic.h"
@implementation Integer(Arithmetic)
-(id) add:(Integer *)addend
{
self.integer= self.integer +addend.integer;
return self;
}
-(id) sub:(Integer *)subtrahend
{
self.integer= self.integer -subtrahend.integer;
return self;
}
@end
Display.h
#import "Integer.h"
@interface Integer(Display)
-(id) showstars;
-(id) showint;
@end
Display.m
#import "Display.h"
@implementation Integer(Display)
-(id) showstars
{
int i, x =self.integer;
for(i=0; i < x; i++)
printf("*");
printf("\n");
return self;
}
-(id) showint
{
printf("%d\n", self.integer);
return self;
}
@end
main.m
#import "Integer.h"
#import "Arithmetic.h"
#import "Display.h"
int
main(void)
{
Integer*num1 = [Integer new], *num2 = [Integer new];
int x;
printf("Enter an integer: ");
scanf("%d",&x);
num1.integer= x;
[num1 showstars];
printf("Enter an integer: ");
scanf("%d",&x);
num2.integer= x;
[num2 showstars];
[num1 add:num2];
[num1 showint];
return 0;
}
11、#import
在C语言中,#include预处理指令总是使被包含的文件内容被插入指令点。在Objective-C中,类似的指令#import保证一个文件只会被包含一次,类似于一般头文件中的
#ifndef XXX
#define XXX ...
#endif
惯用法,或MSVC中的
#pragma once
12、属性
Objective-C 2.0引入了新的语法以声明变量为属性,并包含一可选定义以配置访问方法的生成。属性总是为公共的,其目的为提供外部类访问(也可能为只读)类的内部变量的方法。属性可以被声明为“readonly”,即只读的,也可以提供储存方法包括“assign”,“copy”或“retain”(简单的赋值、复制或增加1引用计数)。默认的属性是原子的,即在访问时会加锁以避免多线程同时访问同一对象,也可以将属性声明为“nonatomic”(非原子的),避免产生锁。
例子:
@interface 类名
{
成员
}
@property 成员;
@end
@implementation类名
@synthesize 成员;
@end
类名 * 对象 =[[类名 alloc]init];
对象.属性=xxx;
@interface Person: NSObject {
@public
NSString*name;
@private
int age;
}
@property(copy)NSString*name;
@property(readonly)int age;
-(id)initWithAge:(int)age;
@end
属性的访问方法由@synthesize关键字来实现,它由属性的声明自动的产生一对访问方法。另外,也可以选择使用@dynamic关键字表明访问方法会由程序员手工提供。
@implementation Person
@synthesize name;
@dynamic age;
-(id)initWithAge:(int)initAge
{
age = initAge; // 注意:直接赋给成员变量,而非属性
return self;
}
-(int)age
{
return 29; // 注意:并非返回真正的年龄
}
@end
属性可以利用传统的消息表达式、点表达式或"valueForKey:"/"setValue:forKey:"方法对来访问。
Person *aPerson = [[Person alloc] initWithAge:53];
aPerson.name= @"Steve";//注意:点表达式,等于[aPerson setName: @"Steve"];
NSLog(@"Accessby message (%@), dot notation(%@), property name(%@) and direct instancevariable access (%@)",
[aPerson name],aPerson.name, [aPerson valueForKey:@"name"], aPerson->name);
为了利用点表达式来访问实例的属性,需要使用“self”关键字:
-(void) introduceMyselfWithProperties:(BOOL)useGetter
{
NSLog(@"Hi, my name is%@.",(useGetter ? self.name : name)); // NOTE: getter vs. ivar access
}
类或协议的属性可以被动态的读取。
int i;
intpropertyCount= 0;
objc_property_t*propertyList = class_copyPropertyList([aPersonclass],&propertyCount);
for( i=0; i < propertyCount; i++){
objc_property_t*thisProperty =propertyList+ i;
const char* propertyName=property_getName(*thisProperty);
NSLog(@"Person has aproperty: '%s'", propertyName);
}
13、快速枚举
比起利用NSEnumerator对象或在集合中依次枚举,Objective-C2.0提供了快速枚举的语法。在Objective-C 2.0中,以下循环的功能是相等的,但性能特性不同。
//使用NSEnumerator
NSEnumerator*enumerator =[thePeople objectEnumerator];
Person *p;
while( (p = [enumeratornextObject])!=nil ) {
NSLog(@"%@ is %i yearsold.",[p name], [p age]);
}
//使用依次枚举
for( int i = 0; i < [thePeople count];i++) {
Person*p = [thePeople objectAtIndex:i];
NSLog(@"%@ is %i yearsold.",[p name], [p age]);
}
//使用快速枚举
for(Person *p in thePeople) {
NSLog(@"%@ is %i yearsold.",[p name], [p age]);
}
快速枚举可以比标准枚举产生更有效的代码,由于枚举所调用的方法被使用NSFastEnumeration协议提供的指针算术运算所代替了。
14、静态变量static修饰的成员
@interface 类名
{
成员
}
@property 成员;
+(void)func;
@end
以+开头的方法 类方法可以通过
[类名 方法名]调用
15、点(.)语法
Student *stu = [[[Studentalloc]init]autorelease];
stu.age = 20; //相当于调用了对象的setAge方法(写)
int age = stu.age;//相当于调用了对象的getAge方法(读)
16、内存管理
1>凡是从NSObject中继承的类都需要自己管理内存,在OC语法中,创建的任何一个对象都拥有一个引用计数器,第一次创建的时候这个引用计数器为1,当引用计数器的值为0时,该对象会被销毁。内存管理涉及到以下接口:
* release :对象的引用计数器减1
* retain :对象的引用计数器加1
* retainCount :获取对象当前引用计数器的数量
对象生命周期回调接口:
* init :对象的默认构造方法,如果自定义构造方法用于初始化成员变量时,必须先调用父类的构造方法,并判断获得的对象是否为nil,再初始化成员变量。
如:- (void)initWithAge:(int)age {
if ( self = [superinit] ) {
_age =age;
}
return self;
}
* dealloc :对象销毁时,系统会自动调用该方法,通常在该方法中释放内存或其它资源。在重写dealloc方法时,注意在代码最后面调用父类的dealloc方法,用于释放内存等相关资源。
如:
- (void)dealloc {
[_book release];//释放成员变量
[super dealloc];
}
2>不需要管理内存的对象
*基本数据类型
*系统自带的类调用自己的静态方法创建的对象,是自动释放内存的,不需要管理
3>内存管理原则
只有向对象发送了alloc,retain,copy,new消息才有必要做release操作
*谁alloc,retain,copy,new谁release
*谁创建谁释放(release)
*谁没有allock,retain,copy,new,你就不要做release操作
4>自动内存管理(由autoreleasepool管理)
在创建对象的同时,调用autorelease方法,会将该对象的一个引用自动存放到最近创建的一个自动释放释放池中。以后该对象就不需要手动来release操作,徐非做了retain,copy等修改了引用计数器的操作。当自动释放池被销毁时,会向池子中所有对象发送一个release消息,池子中的所有对象的引用计数器此时会减1,只有当池子中的引用计数器为0时,该对象才会被彻底销毁。不是说只要将对象交给自动释放池了,池子被销毁,池子中的所有对象就一定会被销毁。
如:@autoreleasepool {
Student *stu = [[[Student alloc]init]autorelease];//此时stu对象将放到这个大括号中的自动释放池子中
[stu retain];//如果加了这一句,此时stu的引用计数器为2,这时候如果在池子销毁前没有向对象发送release消息,就算池子销毁,该对象也还是会造成内存泄露
} //程序执行到此处,代表自动释放池被销毁,意味着池子中的所有对象都会接到一个release消息