使用dis模块的dis()可以显示函数的字节码,它的参数可以是函数对象或者代码对象。
如果是函数对象
>>> dis.dis(myfunc)
2 0 LOAD_GLOBAL 0 (len)
3 LOAD_FAST 0 (alist)
6 CALL_FUNCTION 1
9 RETURN_VALUE
如果是代码对象,就使用
python -m dis test.py
dis输入的文本格式如下
程序中的行号 | 字节码偏移量 | 字节码的命令 | 命令参数 | 参数的相关说明 |
---|---|---|---|---|
4 | 0 | LOAD_FAST | 0 | (a) |
#coding=utf-8
while 1:
pass
while True:
pass
✘ ~/SY/test python -m dis test1.py
3 0 SETUP_LOOP 4 (to 7)
4 >> 3 JUMP_ABSOLUTE 3
6 POP_BLOCK
6 >> 7 SETUP_LOOP 10 (to 20)
>> 10 LOAD_NAME 0 (True)
13 POP_JUMP_IF_FALSE 19
7 16 JUMP_ABSOLUTE 10
>> 19 POP_BLOCK
>> 20 LOAD_CONST 0 (None)
JUMP_ABSOLUTE(目标)
设置字节码计数器到目标。
POP_BLOCK()
从块堆栈中移除一个块。每帧,有一堆块,表示嵌套循环,try语句等
SETUP_LOOP(delta )
将块的块推送到块堆栈。该块从当前的指令跨越大小为delta字节。
LOAD_NAME(namei )
将与之关联的值推co_names[namei]送到堆栈上。
POP_JUMP_IF_FALSE(目标)
如果TOS为false,则将字节码计数器设置为目标。TOS弹出。
根据字节码的解释,可以看出while 1是直接设置计数器,POP_BLOCK在这里我不太清楚,感觉应该是计数器减1,如果不对请大佬指正一下。而while True要现做压栈操作,然后弹出来判断。
而且True在python2中不是一个关键字,只是一个内置的变量,bool类型,值为1,即True+True输出2.而且还可以被赋值.比如赋值True = 2, 甚至可以赋值True = False.
所以while True的时候, 每次循环都要检查True的值, 对应指令LOAD_NAME.这就是为什么while True比while 1慢了.
不过在python3中,True变成了关键字了.while 1和while True的指令相同,所以没有性能区别了.