前言:
本篇为视频学习笔记
我们平时写的是表层代码,最终转成机器码是什么? 它在内存中是怎么操作的?在寄存器CPU中又是怎么操作?我们看不到的,不利于我们掌握编程语言的本质。
我们习惯于网上搜集各类IT教程,但是不知道,它是对是错,最简单的方法就是转成汇编。
举个例子 OC
利用Xcode新建一个名为TestOC项目
我们知道OC中有一东西叫做sizeof,是测量一个类型占用多少个字节,比如我们写一个Int,然后我们在printf打印一下,假设我们这样做:
有的资料说的非常离谱,sizeof是一个函数,因为它看上去是sizeof函数名加一个小括号,因为平时我们调用函数就是先来一个函数名在加上一个小括号传参。举这个例子。以后碰见别的,也是一样的。
究竟sizeof是个什么东西呢?表面上是看不出来的,我们可以在printf位置打一个断点,comd+R,把程序运行一下:
断点打好了,接下来,我们要做些什么事情呢?把它转成汇编。
找到 Debug ---> Debug Workflow---> AlWays Show Disassembly(总是显示反汇编)
反汇编就是把你平常写的代码转成汇编。点进去之后 (如下图):
断点往下是对应的printf有关,上部分是printf前面的
第9行就是 int c = sizeof(int)所对应的汇编代码,也就是 int c = sizeof(int)底层干了什么事情
变量c的内存空间,干了一件什么事情,把这个 符号。也就是说,以美元$开头的都是一个立即数。说白了也就是说编译器看见了sizeof(int)运算符,直接将sizeof(int)变成了4
所以你 int c = sizeof(Int) 相当于 int c = 4,从汇编代码中可以看出来,sizeof不是函数,如果是函数调用,它变成汇编是callq,callq就是调用函数的意思。
上边这个是print函数,如上面的解释。这个0x100000f78内存地址,可以认为是print函数调用的地址。当我们看sizeof的时候,明显没有看到callq,既然没有看到callq就证明它不是一个函数。你一旦使用汇编去分析代码的话,很多本质就会一目了然,不用在相信一些网上的资料了。所以掌握好汇编就是掌握好真理的一种方式,可以真正掌握变成语言的本质,swift 也是一样
Swift例子
同样是如下面代码,我们像窥探一下 var c = a + b的本质,打上断点,comd + R运行一下,会自动转成汇编,因为上面的例子已经设置了AlWays Show Disassembly.
加法是怎么实现的呢?就是断点下面这些代码。后面的代码也有专门的一些解释TestSwift.a : Swift.Int / TestSwift.b : Swift.Int
很奇怪,只是一个简单的加法,按理来说汇编语言中应该看到一个加法的指令。其实,在我们这个汇编中,加法的指令是add后面也许会加单位,比如说addq。但是你会发现,我们看到的a,看到的b。左边好像没有看见add。你可能在下图那个地方看到一个addq.那么我们想,作为一个简单的加法,为什么要搞这么复杂,下部分还牵扯到多个callq。就说明了,一个简单的加法干了些什么事情,不是表面上理解的a + b,转成汇编的话,就会知道它做了好多事情。