一、概述

如上图:

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

GitHub

class和object_getClass方法区别的更多相关文章

  1. querySelectorAll 方法相比 getElementsBy 系列方法区别

    最近有人问到querySelectorAll 方法相比 getElementsBy 系列方法区别,一时没想起来说些什么,今天查下文档,总结一下它们的区别,以便自己理解. 1. W3C 标准queryS ...

  2. 浅析对象访问属性的"."和"[]"方法区别

    在JavaScript中通常使用”."运算符来存取对象的属性的值.或者使用[]作为一个关联数组来存取对象的属性.但是这两种方式有什么区别了? 例如,读取object中的property属性值 ...

  3. JAVAAPI学习之Calendar类;Calendar类set()、add()、roll()方法区别

    JAVAAPI学习之Calendar类 http://blog.csdn.net/myjlvzlp/article/details/8065775(写的很好,清晰易懂) Calendar类set(). ...

  4. java中File类中list()和listFiles()方法区别

    list()和listFiles()方法区别: 1.返回值类型不同:前者为String数组,后者为File对象数组 2.数组中元素内容不同:前者为string类型的[文件名](包含后缀名),后者为Fi ...

  5. jQuery方法区别:click() bind() live() delegate()区别

    今天看到一篇jquery 事件的文章,自己写了个小例子,虽然2种方式都可以实现,但是不太明白,找了点资料 $("#box1").delegate("p",&qu ...

  6. JavaScript toString、String和stringify方法区别

    <!DOCTYPE html> <html> <head> <meta charset="utf-8" /> <title&g ...

  7. Java构造器(构造方法)与方法区别

    构造器,又称为构造方法.构造器用于构造该类的实例,也就是对象. 格式如下:[修饰符]  类名 (形参列表){//n条语句} 构造方法是一种特殊的方法,与一般的方法区别:  1.构造方法的名字必须与定义 ...

  8. DevOps - 与敏捷方法区别

    章节 DevOps – 为什么 DevOps – 与传统方式区别 DevOps – 优势 DevOps – 不适用 DevOps – 生命周期 DevOps – 与敏捷方法区别 DevOps – 实施 ...

  9. 从源码看commit和commitAllowingStateLoss方法区别

    Fragment介绍 在很久以前,也就是我刚开始写Android时(大约在2012年的冬天--),那时候如果要实现像下面微信一样的Tab切换页面,需要继承TabActivity,然后使用TabHost ...

随机推荐

  1. 用CSS的方法如何让一个元素不可见?(面试题目)

    面试中看到这个问题,自己想的不全面,下面整理下,一起学习: 一.CSS元素隐藏 在CSS中,让元素隐藏(指屏幕范围内肉眼不可见)的方法很多,有的占据空间,有的不占据空间:有的可以响应点击,有的不能响应 ...

  2. Eclipse Ctrl + H 搜索文件不覆盖已打开文件解决办法

    1.windows------->preferences

  3. jquery validation表单验证插件。

    这个是刚学的,觉得对以后挺有用的,就想把自己所学的分享一下. 校验规则: (1)required:true 必输字段 (2)number:true 必须输入合法的数字(负数,小数) (3)digits ...

  4. Oracle GoldenGate OGG管理员手册(较早资料)

    第一章 系统实现简述 前言 编写本手册的目的是为系统管理员以及相关操作人员提供 Oracle  Goldengat  软 件的日常维护和使用的技术参考: 3 ORACLE 第二章 OGG 日常维护操作 ...

  5. UNIX高级环境编程(8)进程环境(Process Environment)- 进程的启动和退出、内存布局、环境变量列表

    在学习进程控制相关知识之前,我们需要了解一个单进程的运行环境. 本章我们将了解一下的内容: 程序运行时,main函数是如何被调用的: 命令行参数是如何被传入到程序中的: 一个典型的内存布局是怎样的: ...

  6. spring4声明式事务--01注解方式

    1.在spring配置文件中引入 tx 命名空间 xmlns:tx="http://www.springframework.org/schema/tx" 2.配置事务管理器 < ...

  7. November 5th Week 45th Saturday 2016

    The longest day has an end. 最难过的日子也会有尽头. No, no, no, I can't see the end of such days, of course, if ...

  8. U-Mail详解邮件营销优势及应用领域

    最近频频有营销人员向U-Mail小编咨询:邮件营销到底有什么好处呢?与此同时,还有不少人对邮件营销存在一定的误解:邮件营销是不是只给潜在消费者发送邮件推广商品呢?其实邮件群发的应用面非常广泛,可不仅仅 ...

  9. 学习python第三天数据库day2

    day01回顾: 数据库: 定义:存储数据的仓库(database,简称db) 常用的数据库对象有哪些? 1).数据表(table) ***** 2).视图(view) 3).索引(index) 4) ...

  10. 1812: [Ioi2005]riv

    1812: [Ioi2005]riv Time Limit: 10 Sec Memory Limit: 64 MB Submit: 635 Solved: 388 [Submit][Status][D ...