在第5讲中已经介绍了如何定义类和建立并初始化对象,好比有Student这个类html
#import <Foundation/Foundation.h> @interface Student : NSObject { int _age; } - (void)setAge:(int)age; - (int)age; @end
#import "Student.h" @implementation Student - (void)setAge:(int)age { _age = age; } - (int)age { return _age; } @end
#import "Student.h" int main(int argc, const char * argv[]) { @autoreleasepool { Student *stu = [[Student alloc] init]; stu.age = 10; [stu release]; } return 0; }
* 在第7行调用Student的alloc方法分配内存,而后再调用init方法初始化对象ios
* 像init这样用来初始化对象的方法,咱们能够称为"构造方法"函数
默认的构造方法,也就是init方法,它是不接收任何参数的。所以,在实际开发中,为了方便,会常常自定义构造方法。spa
接下来,自定义一个构造方法,能够传入一个age参数来初始化Student对象code
- (id)initWithAge:(int)age;
* 构造方法的方法名通常都会以init开头,返回值跟init方法同样为id类型,id能够表明任何OC对象orm
* 这个构造方法接收一个int类型的age参数,目的是在初始化Student对象时,顺便设置成员变量_age的值htm
- (id)initWithAge:(int)age { self = [super init]; if (self != nil) { _age = age; } return self; }
* 跟Java同样,构造方法内部首先要调用父类的构造方法,在第2行调用了父类的init方法,它会返回初始化好的Student对象,这里把返回值赋值给了self,self表明Student对象自己对象
* 第3~5行的意思是:若是self不为nil,也就是初始化成功,就给成员变量_age进行赋值blog
* 最后返回初始化事后的self,整个构造方法就结束了ip
因为C语言和OC的语法特性,咱们能够对构造方法进行简化,先简化第3行
- (id)initWithAge:(int)age { self = [super init]; if (self) { _age = age; } return self; }
* 第3行的 if(self) 跟 if(self!=nil) 是等价的
* 还能够将第二、3行合并,继续简化
- (id)initWithAge:(int)age { if ( self = [super init] ) { _age = age; } return self; }
* 第2行的整体意思是:先调用父类的构造方法init,而后将返回值赋值给self,接着判断self是否为nil
* 之后的构造方法都这样写了
1 Student *stu = [[Student alloc] initWithAge:10]; 2 3 NSLog(@"age is %i", stu.age); 4 5 [stu release];
* 在第1行调用了构造方法initWithAge:,并传入10做为参数,所以Student对象的成员变量_age会变为10
* 在第3行打印Student的成员变量_age,打印结果:
2013-04-19 21:36:47.880 构造方法[448:303] age is 10
众所周知,咱们能够用NSLog函数来输出字符串和一些基本数据类
1 int age = 11; 2 NSLog(@"age is %i", age);
* 第2行的%i表明会输出一个整型数据,右边的变量age会代替%i的位置进行输出
* 输出结果:
2013-04-19 21:43:47.674 构造方法[483:303] age is 11
其实,除了能够输出基本数据类型,NSLog函数还能够输出任何OC对象
1 Student *stu = [[Student alloc] initWithAge:10]; 2 3 NSLog(@"%@", stu); 4 5 [stu release];
* 在第3行用NSLog函数输出stu对象,注意左边的格式符%@,之后想输出OC对象,就得用%@这个格式符
* NSLog函数一旦发现用%@输出某个OC对象时,就会调用这个对象的description方法(这个方法返回值是NSString类型,是OC中的字符串类型),而且将description方法返回的字符串代替%@的位置进行输出
* description方法的默认实现是返回这样的格式:<类名: 对象的内存地址>,所以上面代码的输出结果为:
2013-04-19 21:46:49.896 构造方法[492:303] <Student: 0x100109910>
Student是类名,0x100109910是对象的内存地址
* 注意了,%@只能用于输出OC对象,不能输出结构体等其余类型
* 有Java开发经验的人应该能感觉到OC中的description方法就是Java中的toString方法
description方法的默认实现是返回类名和对象的内存地址,这样的话,使用NSLog输出OC对象,意义就不是很大,由于咱们并不关心对象的内存地址,比较关心的是对象内部的一些成变量的值。所以,会常常重写description方法,覆盖description方法的默认实现
好比,重写Student的description方法,返回成员变量_age的值
1 - (NSString *)description { 2 return [NSString stringWithFormat:@"age=%i", _age]; 3 }
* 在第2行调用了NSString这个类的静态方法stringWithFormat初始化一个字符串对象,并返回这个字符串
* 若是你会使用NSLog的话,那就应该能理解第2行的方法参数是什么意思了
* 假如_age是10,那么description方法返回的字符串就是@"age=10"
* 可能有人会以为奇怪,以前建立的Student对象是须要释放的,为何这里建立的字符串对象不用释放?要想完全明白这个问题,须要先了解OC的内存管理,这里咱们暂不作详细讨论,后面会有章节详细讨论内存管理。你能够先记住一个规则:通常状况下,静态方法返回的对象,都不用手动释放。
* 重写完description方法后,再次执行下面的代码
1 Student *stu = [[Student alloc] initWithAge:10]; 2 3 NSLog(@"%@", stu); 4 5 [stu release];
输出结果为:
2013-04-19 22:09:56.625 构造方法[531:303] age=10
千万不要在description方法中同时使用%@和self,下面的写法是错误的:
1 - (NSString *)description { 2 return [NSString stringWithFormat:@"%@", self]; 3 }
第2行同时使用了%@和self,表明要调用self的description方法,所以最终会致使程序陷入死循环,循环调用description方法
注:本文转自M了个J博客。