类的本质——类对象
一段代码:
#import <Foundation/Foundation.h>
@interface Person : NSObject
-(void)run;
+(void)run;
@end
@implementation Person
-(void)run
{
NSLog(@"对象方法run!");
}
+(void)run
{
NSLog(@"类方法run!");
}
@end
int main(int argc, const char * argv[]) {
@autoreleasepool {
Person *p1=[Person new];
[p1 run];
[Person run];
Person *p2=[Person new];
// 其实类也是一个对象,他也有他的类。比如Person其实就是一个对象,我们将这种对象称之为 类对象。而p1我们称之为类类型的实例对象,这两个是不一样,注意叫法区分。
// 类对象属于Class类型
// 类对象的获取方式:
// ①通过实例对象获取:
Class c1=[p1 class];//这两句话都是得到Dog这个类对象,需要注意的是,Class类声明类对象的时候,直接就是Class 变量名,变量名(如c1)前面是没有符号的(这是结构体,不是指针)
Class c2=[p2 class];
// 我们打印检测一下c1、c2的地址如何
NSLog(@"c1=%p",c1);//c1=0x100001200
NSLog(@"c2=%p",c2);//c2=0x100001200
// 显然c1、c2虽为累类型不同的实例对象调用,但他们属于同一个累类型,所以说他们都是返回的Dog这个类对象的地址
// 当然我们还可以用%@的格式查看一下这个类类型的信息,显然输出的结果是 Person
NSLog(@"%@",c1);//Person
NSLog(@"%@",c2);//Person
// ②通过类名获取
Class c3=[Person class];
NSLog(@"c3=%p",c3);
NSLog(@"%@",c3);
}
return 0;
}
———————————————————————————————————————————
类对象的使用
一段代码:
#import <Foundation/Foundation.h>
@interface Person : NSObject
-(void)test;
+(void)test;
@end
@implementation Person
-(void)test
{
NSLog(@"-test!");
}
+(void)test
{
NSLog(@"+test!");
}
@end
int main(int argc, const char * argv[]) {
@autoreleasepool {
// 获取类对象
Class c1=[Person class];//此时c1就等同于Person
// 类对象的使用:
// ①使用类对象创建实例对象
Person *p1=[c1 new];//这句话等同于 Person *p1=[Person new];
// c1 *p1=[c1 new];//★但是我们却不能用这种方法进行创建实例对象,这一点一定要记住!
[p1 test];//用实例对象p1调用对象方法test,调用成功说名创建实例对象成功
// ②使用类对象调用类方法
[c1 test];
}
return 0;
}
———————————————————————————————————————————
类对象的存储及SEL(理解)
在实例对象调用方法的时候,首先在类对象中进行判断(判断当前调用的方法的SEL 和 类的代码区里的SEL相比是不是一致),然后确定是不是调用。
下面介绍一下SEL:
★SEL全称selector表示方法的存储位置★
例如:
Person *p = [ [ Person alloc ] init ] ;
[ p test ] ; //这里test是一个对象方法
寻找方法的过程:
①首先把test这个方法名包装成sel类型的数据
②根据sel数据找到对应的方法地址
③根据方法地址调用相应的方法
(注意一下,这个查找sel数据的过程有缓存,第一次找一个一个找非常的费时且非常耗性能,但是第二次就直接使用了)
关于_cmd: 每个方法的内部都有一个_cmd: ,他代表了当前方法。
★注意:SEL其实是对方法的一种包装,将方法包装成一个SEL类型的数据,去找寻对应方法的地址,找到方法地址后就可以调用方法了。这些都是运行时的特性,发消息就是发送SEL,然后根据SEL找到地址,进而调用方法。
一段代码并不完整(只是测试用的):
// 如果用SEL来调用test方法的话,我们可以这样做:
// 手动将test方法(对象方法)包装成 SEL 类型
SEL s1=@selector(test); //这里和Class声明类对象的时候是一样的,属于结构体类型
[p1 performSelector:s1];
// 上面两句话,和 [p1 test]; 的作用是一模一样的
[Person performSelector:s1]; //performSelector:s1就等同于test
// 当然这里也可以调用类方法
版权声明:本文为博主原创文章,未经博主允许不得转载。