class和object_getClass方法区别
一、概述
如上图:
1.内存创建一个instance实例对象(Person *per),同时会创建一个与之对应的类对象(Class perClass)和元类对象(Class perMeta);
注:实例对象通过calloc可创建多个,但类对象和元类对象在内存中只有一份,只创建一次;
2.对象的本质,其实是C语言的结构体struct,各个对象的内存结构为:
per:isa指针+仅存储Person类成员变量的值;
Person:isa指针+superclass指针+存储成员变量的类型、名称,协议,对象方法等;
perMeta:isa指针+superclass指针+仅存储类方法;
3.isa指向:
per:指向类对象Person;
Person:指向元类对象perMeta;
perMeta:指向基类(Root,如:NSObject)的元类对象meta(基类的元类对象的isa指向该元类对象自己);
4.superclass指向:
Person:指向父类>>基类的类对象指向nil;
perMeta:指向父类>>基类的元类对象指向该基类的类对象;
二、代码分析
1)通过实例对象per获取类对象
- (void)viewDidLoad {
[super viewDidLoad];
self.per1 = [[Person alloc] init];
self.per2 = [[Person alloc] init];
//类对象
Class perClass1 = [self.per1 class];
Class perGetClass2 = object_getClass(self.per1);
Class person = [Person class];
//打断点
NSLog(@"---");
}
进入lldb模式:
//所有的对象(包括类对象和实例对象)所属类的名字均为“Person”
//查看类对象自身地址和self.per成员变量isa的地址值
//p/x:以十六进制输出
如上图说明三个问题:
第一,每个实例对象开辟单独的内存;
第二,同一种类对象仅在内存中开辟一次;
第三,此处class方法和object_getClass无任何区别;
2)通过类对象获取元类对象
增加代码:
//元类对象
Class perMeta1 = [perGetClass2 class];
Class perMeta2 = object_getClass(perGetClass2);
lldb模式:
报错:引用的成员变量isa不是结构体或共同体的成员;
原因:结构体中的isa变量没有暴露出来,从而无法引用;
解决:自定义相同结构体,并将对象强制转换
//添加代码
struct lyb_objc_class {
Class _Nonnull isa;
}; struct lyb_objc_class *perGetClass3 = (__bridge struct lyb_objc_class *)object_getClass(self.per1);
//lldb模式:
说明:
1.perGetClass2和perGetClass3指的是同一个类对象;
2.perMeta1的地址跟perGetClass2和perGetClass3的地址是相同的,说明此时class并没有返回元类对象,依然是类对象;
3.perMeta2的地址和perGetClass3->isa指向的地址相同,说明object_getClass返回的是元类对象;
4.元类对象的类名称和类对象的一样,依然是Person;
3)通过元类对象获取基类的元类对象
//添加代码
struct lyb_objc_class *perMeta3 = (__bridge struct lyb_objc_class *)object_getClass(perGetClass2);
//还是perMeta2
Class rootMeta1 = [perMeta2 class];
//基类(NSObject)的元类对象
Class rootMeta2 = object_getClass(perMeta2);
//lldb模式:
说明:
1.class返回的依然是元类对象自身,object_getClass返回的是基类的元类对象;
2.基类的元类对象的类名跟类对象的一样,为NSObject;
三、结论
当消息对象为实例对象instance时,class与object_getClass返回的对象地址一样;当消息对象为类对象,或元类对象时,class返回的消息对象本身,而object_getClass返回的是下一个对象;
原因:因为class返回的是self,而object_getClass返回的是isa指向的对象;
说明:以上源码查找在GitHub上有演示;
补充:class <=> objc_getClass
//代码
//clang
class和object_getClass方法区别的更多相关文章
- querySelectorAll 方法相比 getElementsBy 系列方法区别
最近有人问到querySelectorAll 方法相比 getElementsBy 系列方法区别,一时没想起来说些什么,今天查下文档,总结一下它们的区别,以便自己理解. 1. W3C 标准queryS ...
- 浅析对象访问属性的"."和"[]"方法区别
在JavaScript中通常使用”."运算符来存取对象的属性的值.或者使用[]作为一个关联数组来存取对象的属性.但是这两种方式有什么区别了? 例如,读取object中的property属性值 ...
- JAVAAPI学习之Calendar类;Calendar类set()、add()、roll()方法区别
JAVAAPI学习之Calendar类 http://blog.csdn.net/myjlvzlp/article/details/8065775(写的很好,清晰易懂) Calendar类set(). ...
- java中File类中list()和listFiles()方法区别
list()和listFiles()方法区别: 1.返回值类型不同:前者为String数组,后者为File对象数组 2.数组中元素内容不同:前者为string类型的[文件名](包含后缀名),后者为Fi ...
- jQuery方法区别:click() bind() live() delegate()区别
今天看到一篇jquery 事件的文章,自己写了个小例子,虽然2种方式都可以实现,但是不太明白,找了点资料 $("#box1").delegate("p",&qu ...
- JavaScript toString、String和stringify方法区别
<!DOCTYPE html> <html> <head> <meta charset="utf-8" /> <title&g ...
- Java构造器(构造方法)与方法区别
构造器,又称为构造方法.构造器用于构造该类的实例,也就是对象. 格式如下:[修饰符] 类名 (形参列表){//n条语句} 构造方法是一种特殊的方法,与一般的方法区别: 1.构造方法的名字必须与定义 ...
- DevOps - 与敏捷方法区别
章节 DevOps – 为什么 DevOps – 与传统方式区别 DevOps – 优势 DevOps – 不适用 DevOps – 生命周期 DevOps – 与敏捷方法区别 DevOps – 实施 ...
- 从源码看commit和commitAllowingStateLoss方法区别
Fragment介绍 在很久以前,也就是我刚开始写Android时(大约在2012年的冬天--),那时候如果要实现像下面微信一样的Tab切换页面,需要继承TabActivity,然后使用TabHost ...
随机推荐
- react 中使用定时器 Timers(定时器)
setTimeout,clearTmeout setInterval,clearInterval 在 class 中 class TimersDemo extends Component { cons ...
- C/S模式下的打印方法
C/S模式使用润乾报表时有两种打印方法(都使用设计器授权) 1. 使用加密狗打印 这种方式需要使用加密狗,适用于客户端较少时 2. 使用api调用打印方法实现打印 ...
- Android自带语音播报+讯飞语音播报封装(直接用)
一.Android自带的语音播报 1.查看是否支持中文,在测试的设备中打开‘设置’ -->找到 '语言和输入法'-->查看语音选项,是否支持中文,默认仅支持英文. 使用如下: public ...
- redis 命令行查看修改配置文件项、配置文件说明
命令行查看修改配置文件项 config get | config set | config rewrite config get requirepass // 获取密码config set requi ...
- C++ 随机数字以及随机数字加字母生成
#include <time.h>#include <sys/timeb.h>void MainWindow::slot_clicked(){ QString strRand; ...
- jsoncpp cmake
(1)下载jsoncpp源码源码地址:https://github.com/open-source-parsers/jsoncpp/tree/0.y.z(2)解压源码 unzip jsoncpp-0. ...
- 一个Interface 继承多个Interface 的总结
我们知道在Java中的继承都是单继承的,就是说一个父类可以被多个子类继承但是一个子类只能有一个父类.但是一个接口可以被不同实现类去实现,这就是我们说的Java中的多态的概念.下面我们再来说一下接口的多 ...
- SQLServer Temp tables 数据疑问
1. 现象 使用Cacti监控,有关于临时表的一个图形 可以看到正在使用的临时表Active Temp Tables的数量非常大,并且在非工作时间,也维持在400个左右.感觉非常奇怪,所以追查下! 2 ...
- Datastage重启服务
使用DS开发job时,偶尔一个Job出现卡死现象,导致工作不能进展下去,有时候需要重启datastage服务才可以. DataStage在正常运行时候有以下两个主要的进程: (1)dsrpcd(DS的 ...
- DataTable拷贝DataRow
一开始,思路是这样的: DataTable dt = SqlHelper.GetTable("select * from Person"); DataTable dt2 = dt. ...