前言
在Objective-C语言中,类的本质是什么?
一、内存的五大区域是什么?
1.栈——存储局部变量
2.堆——存储程序员用malloc,realloc,calloc自己申请的空间,以及类的对象。
3.BSS段——存储未初始化的全局变量,静态变量
4.数据段(常量区)——存储已经初始化的全局变量,静态变量。
5.代码段——存储类的二进制代码,方法的二进制代码
二、类的本质
1.类什么时候被加载到代码段?
类第一次被访问的时候,被加载到代码段,称为“类加载”。
2.类一旦被加载到代码段以后,什么时候回收?
程序结束时回收。
3.类以什么样的形式存储在代码段?
1)前言:任何存储在内存当中的数据,都有一个数据类型。
2)前言:任何在内存中申请的空间,也有自己的类型。
3)在代码段中存储类的空间是什么类型的:Class类型的
4)在代码段中存储类的步骤:
a. 现在代码段中创建一个Class对象。
b. Class是Foundation框架中的一个类,那Class对象是干嘛的呢,是用来存储类的。
c. 将类的信息存储在这个Class对象之中
d. 这个Class对象至少应该有三个属性:
- ClassName 存储类名
- 存储类的属性们
- 存储类的方法们
e. 记住,这是个对象,所以里面有一个iSA指针,这个指针指向这个类的父类。
4.如何拿到存储在代码段中的类对象?
1)第一种方式,调用存储在Person类中的类方法class,例如:
[Person class];
这个方法就会返回Person这个类,在代码段中存储的那个类对象。
然后,用一个指针接收一下,例如:Class c1 ,但是,由于Class是一个自定义数据类型,它在定义时,已经加了,所以这里就不用再加*了,Class的定义如下:
typedef struct objc_class *Class;
所以,这里我们用c1指针直接接收一下:
Class c1 = [Person class];
这时候,c1指针就指向了存储Person类的那个Class对象。
可以打印出c1的地址来看看:
NSLog(@"c1 = %p " , c1 );
2)第二种方式,调用Person类的对象p1的对象方法class,就能拿到这个对象所属的类,在代码段中存储的那个Class对象的指针,例如:
Person *p1 = [Person new];
Class c2 = [p1 class];
NSLog(@"c2 = %p " , c2 ) ;
可以肯定的是c1和c2是一样的,并且跟p1对象的iSA指针指向的地址是一样的。
也可以说,对象中的iSA指针,就是指向存储在代码段中的Class对象的地址。
5.拿到存储在代码段中的类对象的指针之后,有什么用?
1)如何使用类对象,运行如上代码:
Class c1 = [Person class];
之后,c1指的是存储Person类的类对象,其实可以这么认为,这个时候,c1对象就等价于Person类。所以,在使用Person的地方,完全可以使用c1。
2)比如,使用类对象,来调用类的类方法。例如,Person类里有一个类方法:+ (void)sayHi; + (void)sayHi{NSLog(@“大家好,我是Person。。。”);}
这时候,可以使用如下语句调用这个类方法:
[c1 sayHi];
2)使用类对象,来创建对象,例如:
Class c1 = [Person class];
Person *p1 = [c1 new];
这本质上是使用类对象,调用类方法new,创建对象。
3)注意:通过类对象,不能调用对象方法,因为对象方法,需要通过Person类来调用,而你这个c1是Person对象吗,不是,c1是Person类。
总结
这篇文章讲述了类的本质,包括以下几个重点:
1.类,是怎么存储在代码段里面的。类,是以Class对象的形式,存储在代码段之中的。
2.如何拿到存储类的类对象?调用类的类方法class,或者调用对象的对象方法class。
3.拿到之后有神马用?可以通过类对象,调用类的类方法。
最后,附上本篇文章中Person类的定义:
@interface Person : NSObject
{
NSString *_name;
int _age;
}
++ (void)sayHi;//注意,这句话前面只有1个加号,因为这个微博不能输入加号,所以用了两个加号
@end
@implementation Person
++ (void)sayHi//注意,这句话前面只有1个加号,因为这个微博不能输入加号,所以用了两个加号
{
NSLog(@“大家好,我是Person。。。”);
}
@end