在分析源碼時(shí)會碰到一些代碼箫锤,不糾結(jié)它并不會影響我理解源碼,但是細(xì)看一下會發(fā)現(xiàn)它的確有一些精巧角撞,但是當(dāng)了解了它的含義之后情臭,又發(fā)覺它很簡單,不過我還是把它當(dāng)作一個(gè)小知識赴精,一點(diǎn)點(diǎn)積攢在這里。
因此,這篇文章會不斷的隨著我閱讀源碼而更新的
1. 位運(yùn)算向上舍入
Objects/object.c
的_PyObject_NewVar
函數(shù)中硝清,它首先要用_PyObject_VAR_SIZE
計(jì)算該對象所用內(nèi)存,這里有另外一個(gè)宏_Py_SIZE_ROUND_UP(n, a)
转晰,作用是求出n最接近的a的倍數(shù)(前提條件:n是2的冪)
#define _Py_SIZE_ROUND_UP(n, a) (((size_t)(n) + \
(size_t)((a) - 1)) & ~(size_t)((a) - 1))
這里有點(diǎn)太復(fù)雜芦拿,簡化一下就是下面的代碼,簡化的代碼直接使用可能會出問題查邢,但它比較簡潔蔗崎,在理解含義時(shí)看一下是不錯(cuò)的
#define _Py_SIZE_ROUND_UP(n, a) (n + (a-1) & ~(a-1))
舉個(gè)例子,a為4(2的2次冪)扰藕,則a-1的二進(jìn)制是11缓苛,我們舉一些例子,就可以看出它的效果了:(表格中前為二進(jìn)制实胸,后為十進(jìn)制)
n | n + (a-1) | (n + (a-1) & ~(a-1)) |
---|---|---|
100/4 | 111/7 | 100/4 |
101/5 | 1000/8 | 1000/8 |
110/6 | 1001/9 | 1000/8 |
111/7 | 1010/10 | 1000/8 |
代碼重點(diǎn)就在通過& ~(a-1))將后兩位二進(jìn)制變?yōu)?他嫡,而如果不給n加(a-1),那結(jié)果就是向下舍入
2. 巧妙利用||
賦值
Python/pystate.c
中
static PyThread_type_lock head_mutex = NULL; /* Protects interp->tstate_head */
#define HEAD_INIT() (void)(head_mutex || (head_mutex = PyThread_allocate_lock()))
有時(shí)候會碰到當(dāng)一個(gè)變量值為空時(shí)賦值的需要庐完,可能會寫
if(variable == NULL)
variable = ....;
不過其實(shí)可以使用||
的一個(gè)特性讓它更簡潔钢属,由于它只需要一個(gè)條件成立那么就成立,如果第一個(gè)表達(dá)式成立门躯,第二個(gè)表達(dá)式成立與否就不重要了淆党,所以可以不運(yùn)行,所以其實(shí)可以這樣寫
variable || variable = ...;
當(dāng)variable為NULL的時(shí)候就需要去運(yùn)行第二個(gè)表達(dá)式讶凉,所以就做到了變量為空時(shí)賦值染乌,這個(gè)特性還可以拓展的,而且我第一次知道這種辦法是在學(xué)javascript
的時(shí)候