Runtime学习基础之isa指针(终结篇)

755 阅读4分钟

如果您在看这篇博客时候,还没看过我上一篇博客,需要您去耐心的看一下,有助于更快更好的理解这篇博客的内容Runtime学习基础之isa指针

好,我们继续上篇博客继续说:我们先用结构体解决这个问题

,请看下面的代码

这里我们定义一个结构体,里面存储3个参数,我们看一下结果能不能解决这个问题:请看下面的代码截图

这个明显是解决了情况,大家可以尝试其他情况,接下来,我们就引入今天最重要的模块,就是共同体union

首先我们先看我们这个需求怎么用共用体union来解决,这里我直接写结果,等会再一起讨论苹果的源码,它也是这么来解决的,因为我们还知道,位运算效率是非常高的,所以请看下图:

这个也可以很好的解决问题,至于结果我就不截图了,大家可以试试,肯定是可以解决这个问题,另外,我上图的绿色部分是可以注释的,也就是有没有绿色都是成功的,说明,绿色部分就是类似一个注解给给我们自己可以看得清楚.

苹果源码解读

好了,有了上面的基础,相信我们很容易理解苹果的源码了,我们现在去看看.因为它的实现就是objc_object 我们搜索这个就能找到.

我们先来看下苹果的源码,因为我们知道,苹果的arm64以后对isa做了一些优化,之前的class里面只有一个isa指针,从之前我们学习的,一个isa是8bit也就是64位,明显用这么多位存储一个isa是有些浪费,所以就加了很多其他的东西,我们先看源码如下图:

上面的点进去看看截图,也就是箭头指的那个struct结构体里面的ISA_BITFIELD这个

我们是找arm64对应的,看看图上的蓝色的,是不是非常熟悉,跟我们定义的一摸一样的方法,这里我把绿色框框里面的东西是做什么的,全部写出来了,大家可以看看

我们知道一个isa是占用8bit也就是64位,我们看看上面用了多少位1+1+1+33+6+1+1+1+19=64正好是64全部用上了

现在我们来验证之前的一个问题,在这个博客中写到深入探究对象的isa指针指向哪里

请看下面截图

之前的博客是直接写,并没有证明是为什么.现在说一下,

证明获取isa为什么要&ISA_MASK

由这2篇博客,大家知道怎么去取某个位上的数字了吧,比如我现在想知道isa里面的nonpointer是多少,我们是不是很好取出来,我们直接拿到这个这个地址&1是不是就行了?对吧

那我们之前&ISA_MASK掩码大家知道为什么了吧,现在我们来看看这个掩码是多少,我们猜测肯定是前三位是000,后面是33个1,因为是取出33位,好我们去看看是不是 ,先看ISA_MASK是多少

,我们用计算器看一下2进制是多少

这个二进制算出来,对应的正好33个1,所以&这个掩码,正好可以取出isa的cls的地址,这下大家知道为什么要&这个掩码才能取出这个值吧!

那我们现在就看一个我上个博客说的,cls的后三位永远是0

isa的内存地址中,后三位永远是0

其实上面已经证明了,下面我们就来看看(大家记得用真机运行测试比较严谨,因为是arm64),我这边就不用真机了,大家看一下就行了

好了,相信到这里,上篇博客的四个问题都是可以解答的了,对于isa相信大家比之前的理解更深刻了,如果面试官问isa相信你更能答到本质!

拓展

其实我们位运行有很多值,比如我们的枚举值,我们在iOS开发中,我们在传枚举值中的任何一个或者多个,苹果底层是怎么知道我们传入的是哪一个呢?这个就是用到位运算来处理的,具体的大家可以自己写一个demo测试

我们要想更好的学习runtime,我们还要学习class的类结构,也要清楚知道内部是什么样的,这样才能学好runtime,虽然比较枯燥,但是很有用!

接下来博客我会介绍Runtime的class结构,来为更好的学习runtime打好基础

如果觉得我写得对您有所帮助,请关注我,我会持续更新😄