在之前我们学的东西里, 我们一直都在使用实例方法, 那么除了实例方法之外, 还有没有另外一种方法呢? 答案是有的, 除了实例方法, 还有一种方法叫做类方法, 所谓的类方法其实就是以类名开头, 不需要创建对象, 直接调用的一种方法, 下面让我们一起来探究一下~~
例子:
#import <Foundation/Foundation.h> @interface Preson : NSObject + (void)printfClass; @end @implementation Preson + (void)printfClass { NSLog(@"该类是Preson"); } @end int main() { [Preson printfClass]; return 0; }
输出的结果:
Cain:2.第二天 Cain$ cc 05-类方法.m -framework Foundation Cain:2.第二天 Cain$ ./a.out 2015-01-17 10:01:42.427 a.out[13853:1530323] 该类是Preson
在某些场合使用类方法可以提高我们的性能, 让我们的程序可以更好的运行~
PS: 在编译的时候, 类一加载就会分配好存储空间, 所以不需要再创建对象, 就可以直接调用类方法.
但是有一个注意点, 不要使用对象调用类方法, 否则就会报错, 比如:
#import <Foundation/Foundation.h> @interface Preson : NSObject + (void)printfClass; @end @implementation Preson + (void)printfClass { NSLog(@"该类是Preson"); } @end int main() { Preson *p = [Preson new]; [p printfClass]; return 0; }
编译链接:
Cain:2.第二天 Cain$ cc 05-类方法.m -framework Foundation 05-类方法.m:20:8: warning: instance method '-printfClass' not found (return type defaults to 'id') [-Wobjc-method-access] [p printfClass]; ^~~~~~~~~~~ 05-类方法.m:3:12: note: receiver is instance of class declared here @interface Preson : NSObject ^ 1 warning generated.
运行结果:
Cain:2.第二天 Cain$ ./a.out 2015-01-17 10:29:26.968 a.out[13892:1535384] -[Preson printfClass]: unrecognized selector sent to instance 0x7f9f3ac106f0 2015-01-17 10:29:26.969 a.out[13892:1535384] *** Terminating app due to uncaught exception 'NSInvalidArgumentException', reason: '-[Preson printfClass]: unrecognized selector sent to instance 0x7f9f3ac106f0' *** First throw call stack: ( 0 CoreFoundation 0x00007fff9587464c __exceptionPreprocess + 172 1 libobjc.A.dylib 0x00007fff9b68b6de objc_exception_throw + 43 2 CoreFoundation 0x00007fff958776bd -[NSObject(NSObject) doesNotRecognizeSelector:] + 205 3 CoreFoundation 0x00007fff957bea84 ___forwarding___ + 1028 4 CoreFoundation 0x00007fff957be5f8 _CF_forwarding_prep_0 + 120 5 a.out 0x000000010d2d1f2a main + 90 6 libdyld.dylib 0x00007fff955795c9 start + 1 7 ??? 0x0000000000000001 0x0 + 1 ) libc++abi.dylib: terminating with uncaught exception of type NSException Abort trap: 6
这个错误又是和之前的那个错误一样, "你所发送给对象的消息无法识别"
2015-01-17 10:29:26.968 a.out[13892:1535384] -[Preson printfClass]: unrecognized selector sent to instance 0x7f9f3ac106f0
原因其实很简单, 让我们回想一下, 实例方法是以" - "开头的, 而类方法则是以" + "开头的, 它们之间不能相互兼容, 所以使用对象去调用类方法会报错.
同样, 如果用类去调用实例方法, 也会如此:
#import <Foundation/Foundation.h> @interface Preson : NSObject - (void)test; @end @implementation Preson - (void)test { NSLog(@"我是实例方法"); } @end int main() { [Preson test]; return 0; }
编译链接:
05-类方法.m:22:13: warning: class method '+test' not found (return type defaults to 'id') [-Wobjc-method-access] [Preson test]; ^~~~ 1 warning generated.
运行结果:
Cain:2.第二天 Cain$ ./a.out 2015-01-17 10:45:18.310 a.out[13933:1542292] +[Preson test]: unrecognized selector sent to class 0x109a01180 2015-01-17 10:45:18.312 a.out[13933:1542292] *** Terminating app due to uncaught exception 'NSInvalidArgumentException', reason: '+[Preson test]: unrecognized selector sent to class 0x109a01180' *** First throw call stack: ( 0 CoreFoundation 0x00007fff9587464c __exceptionPreprocess + 172 1 libobjc.A.dylib 0x00007fff9b68b6de objc_exception_throw + 43 2 CoreFoundation 0x00007fff958775bd +[NSObject(NSObject) doesNotRecognizeSelector:] + 205 3 CoreFoundation 0x00007fff957bea84 ___forwarding___ + 1028 4 CoreFoundation 0x00007fff957be5f8 _CF_forwarding_prep_0 + 120 5 a.out 0x0000000109a00efd main + 45 6 libdyld.dylib 0x00007fff955795c9 start + 1 7 ??? 0x0000000000000001 0x0 + 1 ) libc++abi.dylib: terminating with uncaught exception of type NSException Abort trap: 6
同样是这个错误:
2015-01-17 10:45:18.310 a.out[13933:1542292] +[Preson test]: unrecognized selector sent to class 0x109a01180
但这个是没有找到对应的类方法错误, 而上面那个是没有找到对应的实例方法错误~~
还有另外一种错误:
#import <Foundation/Foundation.h> @interface Preson : NSObject { int age; } + (void)test; @end @implementation Preson + (void)test { NSLog(@"%d", age); } @end int main() { [Preson test]; return 0; }
编译链接:
Cain:2.第二天 Cain$ cc 05-类方法.m -framework Foundation 05-类方法.m:13:18: error: instance variable 'age' accessed in class method NSLog(@"%d", age); ^ 1 error generated.
之前就说了, 类方法依靠的不是对象, 所以也不能在类方法中访问实例变量(成员变量), 只有实例方法才能访问实例变量.
但实例方法和类方法可以同名, 因为类型不一样, 所以可以同名并存, 比如:
#import <Foundation/Foundation.h> @interface Preson : NSObject + (void)test; - (void)test; @end @implementation Preson + (void)test { NSLog(@"1111"); } - (void)test { NSLog(@"3333"); } @end int main() { [Preson test]; Preson *p = [Preson new]; [p test]; return 0; }
编译链接运行结果:
Cain:2.第二天 Cain$ cc 05-类方法.m -framework Foundation Cain:2.第二天 Cain$ ./a.out 2015-01-17 11:06:49.746 a.out[13980:1551760] 1111 2015-01-17 11:06:49.748 a.out[13980:1551760] 3333
但是为了书写规范, 最好不要写同名的类方法和实例方法.
类方法的好处:
1> 不依赖于对象, 执行效率高, 能使用类方法的时候就尽量使用类方法.
场合:
1> 当方法内部不需要使用到实例变量的时候, 就可以改用类方法.
好了这次就讲到这里, 下次我们继续~~~