一、类的本质
1、类也是个对象
其实类也是一个对象,是Class类型的对象,简称“类对象”
Class类型的定义
typedef struct objc_class *Class;
类名就代表着类对象,每个类只有一个类对象
2、+load和+initialize
+load
在程序启动的时候会加载所有的类和分类,并调用所有类和分类的+load方法
先加载父类,再加载子类;也就是先调用父类的+load,再调用子类的+load
先加载元原始类,再加载分类
不管程序运行过程有没有用到这个类,都会调用+load加载
+initialize
在第一次使用某个类时(比如创建对象等),就会调用一次+initialize方法
一个类只会调用一次+initialize方法,先调用父类的,再调用子类的
3、获取类对象的2种方式
Class c = [Person class]; // 类方法
或者
Person *p = [Person new];
Class c2 = [p class]; // 对象方法
4、类对象调用类方法
Class c = [Person class];
Person *p2 = [c new];
二、description方法
1、-description方法
使用NSLog和%@输出某个对象时,会调用对象的-description方法,并拿到返回值进行输出
2、+ description方法
使用NSLog和%@输出某个类对象时,会调用类对象+description方法,并拿到返回值进行输出
3、description方法
description方法是用于输出方法中输出%@时(输出对象),底层调用的,默认情况下输出的是对象的内存地址,若要输出对象就需要对description方法进行重写;当输出时想把字符串进行拼接可以在description方法中调用stringWithFormat方法。
4、修改NSLog的默认输出
重写-description或者+description方法即可
5、死循环陷阱
如果在-description方法中使用NSLog打印self
三、SEL
1、方法的存储位置
每个类的方法列表都存储在类对象中
每个方法都有一个与之对应的SEL类型的对象
根据一个SEL对象就可以找到方法的地址,进而调用方法
SEL类型的定义
typedef struct objc_selector *SEL;
2、SEL对象的创建
SEL s = @selector(test);
SEL s2 = NSSelectorFromString(@"test");
3、SEL对象的其他用法
// 将SEL对象转为NSString对象
NSString *str = NSStringFromSelector(@selector(test));
Person *p = [Person new];
// 调用对象p的test方法
[p performSelector:@selector(test)];
四、NSLog输出增强
__FILE__ :源代码文件名
__LINE__ :NSLog代码在第几行
_cmd :代表着当前方法的SEL
// 下面的代码会引发死循环
- (void)test {
[self performSelector:_cmd];
}
1.NSLog回顾
大家都知道,我们可以用NSLog函数来输出字符串和一些基本数据类
1 int age = 11;
2 NSLog( @" age is %d", age);
* 第2行的%d代表会输出一个整型数据,右边的变量age会代替%d的位置进行输出
* 输出结果:
ge is 11
2.NSLog输出OC对象其实,除了可以输出基本数据类型,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方法返回的字符串代替%@的位置进行输出
*上面代码的输出结果为:0x100109910>
Student是类名,0x100109910是对象的内存地址
例子:(没有重写OC中的description方法时)description方法的默认实现是返回这样的格式:<类名: 对象的内存地址>
* 注意了,%@只能用于输出OC对象,不能输出结构体等其他类型
3.重写description方法
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"
* 重写完description方法后,再次执行下面的代码
Student *stu = [[Student alloc] initWithAge:10];
2
3 NSLog(@"%@", stu);
4
5 [stu release];
1输出结果为:
age=10
4.description方法的陷阱
千万不要在description方法中同时使用%@和self,下面的写法是错误的:
- (NSString *)description {
2 return [NSString stringWithFormat:@"%@", self];
3 }
1第2行同时使用了%@和self,代表要调用self的description方法,因此最终会导致程序陷入死循环,循环调用description方法
|
|