[TOC]
前言
- 更多OC对象相关文章请关注
01、OC-初探-对象原理-【Alloc init】
02、OC-再探-对象原理-【Alloc init】
03、OC-终探-对象原理-【Alloc init】
- 本文主要围绕以下几点内容展开讨论;
ISA是何时和创建的对象进行绑定的?(ISA最重要的目的就是关联了对象和类)
ISA的成员(关联类)
类在内存里面只有一份如何证明?
类、元类、根元类分别是如何指向的?
类的本质是什么?
对象的本质是什么?
ISA成员
1、根据nonpointer
这个值来判断当前ISA指向的是纯isa指针,还是bits
;
- nonpointer为0: 存isa
- nonpointer为1: 不止是类对象地址,isa包含类信息,对象的引用计数等;
2、has_assoc
关联对象标识符,0没有,1存在;
3、has_cxx_dtor
4、shiftcls
5、magic
、weakly_referenced
、deall0cating
、has_sidetable_rc
6、extra_rc
ISA关联对象与类
- ISA绑定代码
-
ISA
64字节成员(x86_64和__arm64是不同的);
二进制打印
p/t
打印
八进制打印
p/o
打印
十进制打印
p/d
打印
- 如何证明ISA指针指向的是当前alloc的类?
方案1
方案2
1、x/4gx
打印对象的地址值;
2、取出第一位的ISA地址值;
3、获取当前创建对象的class
4、用当前对象的地址值&ISA_MASK也就是0x00007ffffffffff8
5、最终得到该地址值和第一位的ISA获取的地址值是一致的,这就验证了ISA指向的就是当前alloc的类;
- 代码走向;
object_getClass
-->obj->getIsa();
-->ISA();
-->return (Class)(isa.bits & ISA_MASK);
子网掩码或者位运算的目的都是为了让某一段内存进行展示
类在内存里面只有一份如何证明?
思考
:子类的ISA和父类的ISA是不是同一个ISA?
答案
:肯定不是一个啊!
元类&根元类
类
:代码写出来的-->内存只有一份-->不是手动个创建的是系统自己创建的;
元类
:系统编译的-->发现了系统有这样一个类-->系统也同时创建了编译器;
对象ISA
-->类
图解
获取类的方法
object_getClass(object)
对象ISA
-->类
--类ISA
>元类
解释:对象是由类实例化创建的,那类也是由元类实例化创建的。
对象ISA
-->类
-->元类
图解
对象ISA
-->类
-->元类
-->根元类
图解
对象ISA
-->类
-->元类
-->根元类
-->根元类
图解
- 第一种情况,当创建的对象是NSObject时;
- 第二种情况,当创建的对象是NSObject的子类时;
- 总结:元类和根元类在创建的对象是NSObject时候可能是相同的,但是当创建的对象是NSObject的子类的时候元类和根元类就不是相同的,主要是根据集成的层级管理来决定元类、根元类、根根元类的具体内容;层级越多代表创建的根元类越多;
ISA走位
要点
1、图中要分为两个维度去看(superclass继承链、ISA走位链)
2、继承链
(子类-->父类-->NSObject类-->nil
找方法找不到才指向nil)
3、ISA走位链
(对象ISA
-->类
-->元类
-->根元类
-->根元类
)
4、根元类也继承与NSObject
类的本质
xoce快捷键:command
+ shift
+ 0
进入xcode
帮助模式;
官网地址:Type Encodings官网地址
类(NSObject)是什么时候创建的?
答案:NSObject
这个基类在程序启动的时候他就创建了, 在编译器创建的。
类的本质是什么?
答案:类的本质就是oject_class,是一个结构体!
objc_class内部的结构是什么样子的?
对象的本质
对象的本质是什么?
答案:
对象的本质就是结构体!
- 验证对象的本质是结构体-步骤1:
1、思考,我们写的.m文件在xcode编译的过程中变成了什么?
2、创建一个空的工程, show in finder到.m文件到目录,打开终端;执行命令; main.m是当前.m文件的名称,main3.cpp是即将生成文件的名称;
$ clang -x objective-c -rewrite-objc main.m -o main3.cpp
3、打开main3.cpp文件;
4、找到该对象名对应的结构体截图;
疑问:默认的成员里面的struct NSObject_IMPL 里面有什么成员?
回答:
struct NSObject_IMPL 里面的成员变量是 isa
(见下图)
5、结构体中的NSObject_IVARS成员是继承来的;其他的成员变量或者属性是我们声明出来的;
成员变量和属性在编译的过程当中有什么区别?
答案:
成员变量没有set、get方法,属性有set、get的方法;
结论:
属性 = 成员变量 + get方法 + set方法;
备注:如果在使用终端clang命令报错,报错内容是"缺少UIKit库的问题,请执行下面指令到控制台即可"
总结
ISA是何时和创建的对象进行绑定的?
答案:ISA最重要的目的就是关联了对象和类 isa <-> cls
;
如何获取一个类的父类?
步骤1
object_getClass
方法即可;
Class object_getClass(id obj)
{
if (obj) return obj->getIsa();
else return Nil;
}
步骤2
getIsa
方法即可;
objc_object::getIsa()
{
if (!isTaggedPointer()) return ISA();
uintptr_t ptr = (uintptr_t)this;
if (isExtTaggedPointer()) {
uintptr_t slot =
(ptr >> _OBJC_TAG_EXT_SLOT_SHIFT) & _OBJC_TAG_EXT_SLOT_MASK;
return objc_tag_ext_classes[slot];
} else {
uintptr_t slot =
(ptr >> _OBJC_TAG_SLOT_SHIFT) & _OBJC_TAG_SLOT_MASK;
return objc_tag_classes[slot];
}
}
步骤3
ISA()
方法即可;
objc_object::ISA()
{
assert(!isTaggedPointer());
#if SUPPORT_INDEXED_ISA
if (isa.nonpointer) {
uintptr_t slot = isa.indexcls;
return classForIndex((unsigned)slot);
}
return (Class)isa.bits;
#else
return (Class)(isa.bits & ISA_MASK);
#endif
}
-
关键代码
return (Class)(isa.bits & ISA_MASK);
isa.bits就是当前isa指向的地址值,ISA_MASK就是掩码,这个位置可以看本文ISA关联对象与类
的节点;
ISA指向
对象ISA
-->类
-->元类
-->根元类
-->根元类
class的继承管理
子类subclass
-->父类superclass
-->NSObject
-->nil
根元类
的父类
是NSObject
类的本质是什么?
结构体;
对象的本质是什么?
结构体;