一、分类-Category
1.思考:如何在不改变原来类的模型的前提下,扩展分类的方法?
两种方式:
1》继承:子类可继承父类原有的方法,并且增加自己的方法.
2》分类; 可以给某个类扩 充方法(不修改原来的代码)
2.分类的使用格式
1》分类的声明
1 @interface 类名 (分类名称) 2 3 // 方法声明 4 5 @end
2》分类的实现
1 @implementation类名 (分类名称) 2 3 // 方法实现 4 5 @end
分类使用示例:
给NSString类添加一个类方法,计算某字符串中阿拉伯数字的个数
#import <Foundation/Foundation.h> @interface NSString (number) /* 给NSString类添加一个类方法,计算某字符串中阿拉伯数字的个数 */ + (int)numberCountOfString:(NSString *)str; /* 对象方法 自己的数字个数不用参数了*/ - (int)numberCount; @end
#import "NSString+number.h" @implementation NSString (number) + (int)numberCountOfString:(NSString *)str { int count = 0; // if (nil == str) // { // return 0; // } // for (int i = 0;i < str.length; i++ ) // { // unichar c = [str characterAtIndex:i]; // if ((c >= '0') && (c <= '9')) // count++; // } count = [str numberCount]; return count; } - (int)numberCount; { int count = 0; for (int i = 0; i < self.length;i++) { unichar c = [self characterAtIndex:i]; if ((c >= '0') && (c <= '9')) count++; } return count; } @end
代码测试:
1 int main() 2 { 3 // 扩展类方法的调用 4 int count1 = [NSString numberCountOfString:@"1fh2hg3 h456d78"]; 5 NSLog(@"1fh2hg3 h456d78 has %d num ",count1); 6 7 // 分析:这个方法不好还得用类的形式调用,改进用对象方法 8 // 扩展对象方法的调用 9 int count2 = [@"123asdfg456" numberCount]; 10 NSLog(@"123asdfg456 has %d num ",count2); 11 12 return 0; 13 }
2015-03-23 14:53:39.922 catgory分类[1042:96183] 1fh2hg3 h456d78 has 8 num
2015-03-23 14:53:39.922 catgory分类[1042:96183] 123asdfg456 has 6 num
分类总结:
2. 分类方法实现中可以访问成员变量
3. 分类可以重新实现原来类中的方法,但会覆盖掉原来的方法,导致原来的方法中没法使用
4. 方法调用优先级:分类(后编译的)-》原来类-》父类
二 .类的本质
Class类型的定义 typedefstruct objc_class *Class;
类名就代表着类对象,每个类只有一个类对象。
2.通过代码类理解类的本质
Person类的声明,声明了一个类方法
1 2 #import <Foundation/Foundation.h> 3 4 @interface Person : NSObject 5 { 6 int _age; 7 } 8 @property int age; 9 10 + (void)test; 11 12 @end 13
Person类的实现
1 2 3 #import "Person.h" 4 5 @implementation Person 6 + (void)test 7 { 8 NSLog(@"调用了类的类方法test方法"); 9 } 10 11 @end
主函数:
1 int main() 2 { 3 4 /* p1 p2 都指向创建的Person类的对象*/ 5 Person *p1 = [[Person alloc] init]; 6 Person *p2 = [[Person alloc] init]; 7 8 9 /* 获取内存中类对象的两种方式 都返回Person这个类在内存中的地址 */ 10 11 // 1.调用对象的class 方法 12 Class c = [p1 class]; 13 Class c2 = [p2 class]; 14 15 // 2.调用类的class方法 16 Class c3 = [Person class]; 17 18 NSLog(@"c = %p c2 = %p c3 =%p",c,c2,c3); 19 // 结果c1 = c2 = c3 说明同一个类在内存中只有一个 20 21 /* 类名就是类对象 和class返回的类对象是一样的 */ 22 [Person test]; 23 [c3 test]; 24 25 // 类对象 = 类名 也可以创建对象 26 Person *p3 = [[c alloc] init]; 27 p3.age = 100; 28 NSLog(@"p3 age %d",p3.age); 29 30 return 0; 31 }
2015-03-23 15:22:22.457类的本质[1109:104403] c = 0x1000025d8 c2 = 0x1000025d8 c3 =0x1000025d8
2015-03-23 15:22:22.457类的本质[1109:104403]调用了类的类方法test方法
2015-03-23 15:22:22.458类的本质[1109:104403]调用了类的类方法test方法
2015-03-23 15:22:22.458类的本质[1109:104403] p3 age 100
+load和+initialize方法的调用过程代码示例:
1 #import "Person.h" 2 3 @implementation Person 4 + (void)test 5 { 6 NSLog(@"调用了类的类方法test方法"); 7 } 8 /* 重写了load方法:只要程序运行就会加载一次仅此一次 */ 9 + (void)load 10 { 11 NSLog(@"Person load!!"); 12 } 13 /* 重写了initialize方法:只在用到的时候会调用*/ 14 + (void)initialize 15 { 16 NSLog(@"Person initialize!!"); 17 } 18 @end
#import "Person.h" @interface Person (loadTest) @end
1 #import "Person+loadTest.h" 2 3 @implementation Person (loadTest) 4 /* 重写了load方法:只要程序运行就会加载一次仅此一次 */ 5 + (void)load 6 { 7 NSLog(@"Person+loadTest load!!"); 8 } 9 /* 重写了initialize方法:只在用到的时候会调用*/ 10 + (void)initialize 11 { 12 NSLog(@"Person+loadTest initialize!!"); 13 } 14 @end
2 /* Student类的声明 继承于Person类*/ 3 @interface Student : Person 4 5 @end
1 #import "Student.h" 2 3 @implementation Student 4 /* 重写了load方法:只要程序运行就会加载一次仅此一次 */ 5 6 + (void)load 7 { 8 NSLog(@"Student load!!"); 9 } 10 /* 重写了initialize方法:只在用到的时候会调用*/ 11 + (void)initialize 12 { 13 NSLog(@"Student initialize!!"); 14 } 15 @end
+load和+initialize方法的调用过程测试代码:
1 int main() 2 { 3 return 0; 4 }
2015-03-23 15:42:40.415类的本质[1246:110623] Person load!!
2015-03-23 15:42:40.417类的本质[1246:110623] Student load!!
2015-03-23 15:42:40.417类的本质[1246:110623] Person+loadTest load!!
从打印的结果可见:不管程序运行过程有没有用到这个类,都会调用+load加载进内存,加载的顺序是,先加载父类在加载子类最后加载分类,只加载一次。
2.程序运行过程中只使用Person这个类,查看调用的过程?
1 int main() 2 { 3 // 只是用Person这个类 4 [[Person alloc] init]; 5 return 0; 6 }
2015-03-23 15:47:35.384 类的本质[1254:112282] Person load!!
2015-03-23 15:47:35.385 类的本质[1254:112282] Student load!!
2015-03-23 15:47:35.385 类的本质[1254:112282] Person+loadTest load!!
2015-03-23 15:47:35.386 类的本质[1254:112282] Person+loadTest initialize!!
从打印的结果可见:加载过程同上,当使用了这个Person类时,优先使用分类的initialize方法不会加载原类的initlize方法,由于没使用子类并不会初始化子类。
3.使用子类
1 2 3 int main() 4 { 5 //程序启动时就会加载项目中所有的类,不管用不用 6 // 使用Student类 7 [[Student alloc] init]; 8 return 0; 9 }
2015-03-23 15:52:42.834 类的本质[1268:113796] Person load!!
2015-03-23 15:52:42.836 类的本质[1268:113796] Student load!!
2015-03-23 15:52:42.836 类的本质[1268:113796] Person+loadTest load!!
2015-03-23 15:52:42.837 类的本质[1268:113796] Person+loadTest initialize!!
2015-03-23 15:52:42.837 类的本质[1268:113796] Student initialize!!
从打印的结果可见:四.SEL数据
1. 方法在内存中的存储:
每个方法在内存中都有一个SEL数据和它对应虚拟成下面的形式
SEL s1 = +test 地址
SEL s2 = -test 地址
2.Objective-C中调用函数的方法是“消息传递”,是利用performSelector传递一个sel类型的消息,再根据sel消息找到方法的地址进而去调用这个方法。
3. SEL类型的定义
typedef struct objc_selector *SEL;
4. SEL对象的创建SEL s = @selector(test);
SEL s2 = NSSelectorFromString(@"test");
SEL使用示例:1 #import <Foundation/Foundation.h> 2 3 @interface Person : NSObject 4 /* 类里面包含方法列表 */ 5 // 每个方法在内存中都有一个sel和他对应 6 + (void)test1; 7 - (void)test2; 8 - (void)test3:(NSString *)str; 9 @end
Person类的实现:
1 #import "Person.h" 2 3 @implementation Person 4 + (void)test1 5 { 6 NSLog(@"test1"); 7 } 8 - (void)test2 9 { 10 // _cmd代表本方法 11 // NSStringFromSelector方法接收sel类型的数据,并将其转化为字符串 12 NSString *str = NSStringFromSelector(_cmd); 13 /* 打印本方法名 */ 14 NSLog(@"In %@ function",str); 15 } 16 - (void)test3:(NSString *)str 17 { 18 NSLog(@"In test3 传给我的字符串是 %@",str); 19 } 20 @end
1 int main() 2 { 3 4 Person *p = [[Person alloc] init]; 5 6 // 直接调用test2方法 7 [p test2]; 8 9 // 使用SEL调用方法的过程 10 // 1.把test2包装成SEL类型的数据 11 // 2.根据SEL数据找到对应的方法地址 12 // 3.根据方法地址调用对应的方法 13 14 [p performSelector:@selector(test2)]; 15 16 /* 方法名字符串调用方法 */ 17 NSString *name = @"test2"; 18 // 把字符串变成SEL 19 SEL s2 = NSSelectorFromString(name); 20 [p performSelector:s2]; 21 22 23 /* 带参数的方法调用 */ 24 // 方式一 25 [p performSelector:@selector(test3:) withObject:@"hello sel"]; 26 27 // 方式二 28 SEL s3 = @selector(test3:); 29 [p performSelector:s3 withObject:@"123"]; 30 31 return 0; 32 }
2015-03-23 16:31:30.853 sel[1368:124580] In test2 function
2015-03-23 16:31:30.854 sel[1368:124580] In test2 function
2015-03-23 16:31:30.854 sel[1368:124580] In test2 function
2015-03-23 16:31:30.854 sel[1368:124580] In test3传给我的字符串是 hello sel
2015-03-23 16:31:30.855 sel[1368:124580] In test3传给我的字符串是 123
sel总结;1 每个类的方法列表都存储在类对象中
2 每个方法都有一个与之对应的SEL类型的对象
3. 根据一个SEL对象就可以找到方法的地址,进而调用方法
4. 使用调用方法需要performSelector传递一个sel类型消息。5. _cmd :代表着当前方法的SEL
五.description方法
1. -description方法
使用NSLog和%@输出某个对象时,会调用对象的-description方法,并拿到返回值进行输出
2. + description方法
使用NSLog和%@输出某个类对象时,会调用类对象+description方法,并拿到返回值进行输出