自拍偷在线精品自拍偷,亚洲欧美中文日韩v在线观看不卡

Python源碼理解: '+=' 和 'xx = xx + xx' 的區(qū)別

開(kāi)發(fā) 后端
+=實(shí)際上是干嘛了: 它應(yīng)該能算是一個(gè)加強(qiáng)版的+, 因?yàn)樗?多了一個(gè)寫(xiě)回本身的功能.不過(guò)是否能夠?qū)懟乇旧? 還是得看對(duì)象自身是否支持, 也就是說(shuō)是否具備Py_NotImplemented標(biāo)識(shí), 是否支持sq_inplace_concat, 如果具備, 才能實(shí)現(xiàn), 否則, 也就是和 + 效果一樣而已。

[[200486]]

前菜

在我們使用Python的過(guò)程, 很多時(shí)候會(huì)用到+運(yùn)算, 例如:

  1. a = 1 + 2 
  2.  
  3. print a 
  4.  
  5. # 輸出 
  6.  

 

不光在加法中使用, 在字符串的拼接也同樣發(fā)揮這重要的作用, 例如:

  1. a = 'abc' + 'efg' 
  2.  
  3. print a 
  4.  
  5. # 輸出 
  6.  
  7. abcefg 

 

同樣的, 在列表中也能使用, 例如:

  1. a = [1, 2, 3] + [4, 5, 6] 
  2.  
  3. print a 
  4.  
  5. # 輸出 
  6.  
  7. [1, 2, 3, 4, 5, 6] 

 

為什么上面不同的對(duì)象執(zhí)行同一個(gè)+會(huì)有不同的效果呢? 這就涉及到+的重載, 然而這不是本文要討論的重點(diǎn), 上面的只是前菜而已~~~

正文

先看一個(gè)例子:

  1. num = 123 
  2.  
  3. num = num + 4 
  4.  
  5. print num 
  6.  
  7. # 輸出 
  8.  
  9. 127 

 

這段代碼的用途很明確, 就是一個(gè)簡(jiǎn)單的數(shù)字相加, 但是這樣似乎很繁瑣, 一點(diǎn)都Pythonic, 于是就有了下面的代碼:

  1. num = 123 
  2.  
  3. num += 4 
  4.  
  5. print num 
  6.  
  7. # 輸出 
  8.  
  9. 127 

 

哈, 這樣就很Pythonic了! 但是這種用法真的就是這么好么? 不一定. 看例子:

  1. # coding: utf8 
  2.  
  3. l = [1, 2] 
  4.  
  5. l = l + [3, 4] 
  6.  
  7. print l 
  8.  
  9. # 輸出 
  10.  
  11. [1, 2, 3, 4] 
  12.  
  13. ------------------------------------------ 
  14.  
  15. l = [1, 2] 
  16.  
  17. l += [3, 4] # 列表的+被重載了, 左右操作數(shù)必須都是iterable對(duì)象, 否則會(huì)報(bào)錯(cuò) 
  18.  
  19. print l 
  20.  
  21. # 輸出 
  22.  
  23. [1, 2, 3, 4] 

 

看起來(lái)結(jié)果都一樣嘛~, 但是真的一樣嗎? 我們改下代碼再看下:

  1. # coding: utf8 
  2.  
  3. l = [1, 2] 
  4.  
  5. print 'l之前的id: ', id(l) 
  6.  
  7. l = l + [3, 4] 
  8.  
  9. print 'l之后的id: ', id(l) 
  10.  
  11.   
  12.  
  13. # 輸出 
  14.  
  15. l之前的id:  40270024 
  16.  
  17. l之后的id:  40389000 
  18.  
  19.   
  20.  
  21. ------------------------------------------ 
  22.  
  23.   
  24.  
  25. l = [1, 2] 
  26.  
  27. print 'l之前的id: ', id(l) 
  28.  
  29. l += [3, 4]  # 列表的+被重載了, 左右操作數(shù)必須都是iterable對(duì)象, 否則會(huì)報(bào)錯(cuò) 
  30.  
  31. print 'l之后的id: ', id(l) 
  32.  
  33.   
  34.  
  35. # 輸出 
  36.  
  37. l之前的id:  40270024 
  38.  
  39. l之后的id:  40270024 

 

看到結(jié)果了嗎? 雖然結(jié)果一樣, 但是通過(guò)id的值表示, 運(yùn)算前后, 第一種方法對(duì)象是不同的了, 而第二種還是同一個(gè)對(duì)象! 為什么會(huì)這樣?

結(jié)果分析

先來(lái)看看字節(jié)碼:

  1. [root@test1 ~]# cat 2.py 
  2.  
  3. # coding: utf8 
  4.  
  5. l = [1, 2] 
  6.  
  7. l = l + [3, 4] 
  8.  
  9. print l 
  10.  
  11.   
  12.  
  13.   
  14.  
  15. l = [1, 2] 
  16.  
  17. l += [3, 4]   
  18.  
  19. print l 
  20.  
  21. [root@test1 ~]# python -m dis 2.py 
  22.  
  23.   2           0 LOAD_CONST               0 (1) 
  24.  
  25.               3 LOAD_CONST               1 (2) 
  26.  
  27.               6 BUILD_LIST               2 
  28.  
  29.               9 STORE_NAME               0 (l) 
  30.  
  31.   
  32.  
  33.   3          12 LOAD_NAME                0 (l) 
  34.  
  35.              15 LOAD_CONST               2 (3) 
  36.  
  37.              18 LOAD_CONST               3 (4) 
  38.  
  39.              21 BUILD_LIST               2 
  40.  
  41.              24 BINARY_ADD           
  42.  
  43.              25 STORE_NAME               0 (l) 
  44.  
  45.   
  46.  
  47.   4          28 LOAD_NAME                0 (l) 
  48.  
  49.              31 PRINT_ITEM           
  50.  
  51.              32 PRINT_NEWLINE       
  52.  
  53.   
  54.  
  55.   7          33 LOAD_CONST               0 (1) 
  56.  
  57.              36 LOAD_CONST               1 (2) 
  58.  
  59.              39 BUILD_LIST               2 
  60.  
  61.              42 STORE_NAME               0 (l) 
  62.  
  63.   
  64.  
  65.   8          45 LOAD_NAME                0 (l) 
  66.  
  67.              48 LOAD_CONST               2 (3) 
  68.  
  69.              51 LOAD_CONST               3 (4) 
  70.  
  71.              54 BUILD_LIST               2 
  72.  
  73.              57 INPLACE_ADD         
  74.  
  75.              58 STORE_NAME               0 (l) 
  76.  
  77.   
  78.  
  79.   9          61 LOAD_NAME                0 (l) 
  80.  
  81.              64 PRINT_ITEM           
  82.  
  83.              65 PRINT_NEWLINE       
  84.  
  85.              66 LOAD_CONST               4 (None) 
  86.  
  87.              69 RETURN_VALUE 

 

在上訴的字節(jié)碼, 我們著重需要看的是兩個(gè): BINARY_ADD 和 INPLACE_ADD!

很明顯:

  • l = l + [3, 4, 5]    這種背后就是BINARY_ADD
  • l += [3, 4, 5]     這種背后就是INPLACE_ADD

深入理解

雖然兩個(gè)單詞差很遠(yuǎn), 但其實(shí)兩個(gè)的作用是很類(lèi)似的, 最起碼前面一部分是, 為什么這樣說(shuō), 請(qǐng)看源碼:

  1. # 取自ceva.c 
  2.  
  3. # BINARY_ADD 
  4.  
  5. TARGET_NOARG(BINARY_ADD) 
  6.  
  7.         { 
  8.  
  9.             w = POP(); 
  10.  
  11.             v = TOP(); 
  12.  
  13.             if (PyInt_CheckExact(v) && PyInt_CheckExact(w)) {    // 檢查左右操作數(shù)是否 int 類(lèi)型 
  14.  
  15.                 /* INLINE: int + int */ 
  16.  
  17.                 register long a, b, i; 
  18.  
  19.                 a = PyInt_AS_LONG(v); 
  20.  
  21.                 b = PyInt_AS_LONG(w); 
  22.  
  23.                 /* cast to avoid undefined behaviour 
  24.  
  25.                    on overflow */ 
  26.  
  27.                 i = (long)((unsigned long)a + b); 
  28.  
  29.                 if ((i^a) < 0 && (i^b) < 0) 
  30.  
  31.                     goto slow_add; 
  32.  
  33.                 x = PyInt_FromLong(i); 
  34.  
  35.             } 
  36.  
  37.             else if (PyString_CheckExact(v) && 
  38.  
  39.                      PyString_CheckExact(w)) {                   // 檢查左右操作數(shù)是否 string 類(lèi)型 
  40.  
  41.                 x = string_concatenate(v, w, f, next_instr); 
  42.  
  43.                 /* string_concatenate consumed the ref to v */ 
  44.  
  45.                 goto skip_decref_vx; 
  46.  
  47.             } 
  48.  
  49.             else { 
  50.  
  51.               slow_add:                                          // 兩者都不是, 請(qǐng)走這里~ 
  52.  
  53.                 x = PyNumber_Add(v, w); 
  54.  
  55.             } 
  56.  
  57.            ...(省略) 
  58.  
  59.   
  60.  
  61.   
  62.  
  63. # INPLACE_ADD 
  64.  
  65. TARGET_NOARG(INPLACE_ADD) 
  66.  
  67.         { 
  68.  
  69.             w = POP(); 
  70.  
  71.             v = TOP(); 
  72.  
  73.             if (PyInt_CheckExact(v) && PyInt_CheckExact(w)) {   // 檢查左右操作數(shù)是否 int 類(lèi)型 
  74.  
  75.                 /* INLINE: int + int */ 
  76.  
  77.                 register long a, b, i; 
  78.  
  79.                 a = PyInt_AS_LONG(v); 
  80.  
  81.                 b = PyInt_AS_LONG(w); 
  82.  
  83.                 i = a + b; 
  84.  
  85.                 if ((i^a) < 0 && (i^b) < 0) 
  86.  
  87.                     goto slow_iadd; 
  88.  
  89.                 x = PyInt_FromLong(i); 
  90.  
  91.             } 
  92.  
  93.             else if (PyString_CheckExact(v) && 
  94.  
  95.                      PyString_CheckExact(w)) {                 // 檢查左右操作數(shù)是否 string 類(lèi)型 
  96.  
  97.                 x = string_concatenate(v, w, f, next_instr); 
  98.  
  99.                 /* string_concatenate consumed the ref to v */ 
  100.  
  101.                 goto skip_decref_v; 
  102.  
  103.             } 
  104.  
  105.             else { 
  106.  
  107.               slow_iadd:                           
  108.  
  109.                 x = PyNumber_InPlaceAdd(v, w);                 // 兩者都不是, 請(qǐng)走這里~ 
  110.  
  111.             } 
  112.  
  113.            ... (省略) 

 

從上面可以看出, 不管是BINARY_ADD 還是INPLACE_ADD, 他們都會(huì)有如下相同的操作:

檢查是不是都是`int`類(lèi)型, 如果是, 直接返回兩個(gè)數(shù)值相加的結(jié)果

檢查是不是都是`string`類(lèi)型, 如果是, 直接返回字符串拼接的結(jié)果

因?yàn)閮烧叩男袨檎娴暮茴?lèi)似, 所以在這著重講INPLACE_ADD, 對(duì)BINARY_ADD感興趣的童鞋可以在源碼文件: abstract.c, 搜索: PyNumber_Add.實(shí)際上也就少了對(duì)列表之類(lèi)對(duì)象的操作而已.

那我們接著繼續(xù), 先貼個(gè)源碼:

  1. PyObject * 
  2.  
  3. PyNumber_InPlaceAdd(PyObject *v, PyObject *w) 
  4.  
  5.  
  6.     PyObject *result = binary_iop1(v, w, NB_SLOT(nb_inplace_add),     
  7.  
  8.                                    NB_SLOT(nb_add)); 
  9.  
  10.     if (result == Py_NotImplemented) { 
  11.  
  12.         PySequenceMethods *m = v->ob_type->tp_as_sequence; 
  13.  
  14.         Py_DECREF(result); 
  15.  
  16.         if (m != NULL) { 
  17.  
  18.             binaryfunc f = NULL
  19.  
  20.             if (HASINPLACE(v)) 
  21.  
  22.                 f = m->sq_inplace_concat; 
  23.  
  24.             if (f == NULL
  25.  
  26.                 f = m->sq_concat; 
  27.  
  28.             if (f != NULL
  29.  
  30.                 return (*f)(v, w); 
  31.  
  32.         } 
  33.  
  34.         result = binop_type_error(v, w, "+="); 
  35.  
  36.     } 
  37.  
  38.     return result; 

 

INPLACE_ADD本質(zhì)上是對(duì)應(yīng)著abstract.c文件里面的PyNumber_InPlaceAdd函數(shù), 在這個(gè)函數(shù)中, 首先調(diào)用binary_iop1函數(shù), 然后進(jìn)而又調(diào)用了里面的binary_op1函數(shù), 這兩個(gè)函數(shù)很大一個(gè)篇幅, 都是針對(duì)ob_type->tp_as_number, 而我們目前是list, 所以他們的大部分操作, 都和我們的無(wú)關(guān). 正因?yàn)闊o(wú)關(guān), 所以這兩函數(shù)調(diào)用最后, 直接返回Py_NotImplemented, 而這個(gè)是用來(lái)干嘛, 這個(gè)有大作用, 是列表相加的核心所在!

因?yàn)閎inary_iop1的調(diào)用結(jié)果是Py_NotImplemented, 所以下面的判斷成立, 開(kāi)始尋找對(duì)象(也就是演示代碼中l(wèi)對(duì)象)的ob_type->tp_as_sequence屬性.

因?yàn)槲覀兊膶?duì)象是l(列表), 所以我們需要去PyList_type需找真相:

  1. # 取自: listobject.c 
  2.  
  3. PyTypeObject PyList_Type = { 
  4.  
  5.     ... (省略) 
  6.  
  7.     &list_as_sequence,                          /* tp_as_sequence */ 
  8.  
  9.     ... (省略) 
  10.  

 

可以看出, 其實(shí)也就是直接取list_as_sequence, 而這個(gè)是什么呢? 其實(shí)是一個(gè)結(jié)構(gòu)體, 里面存放了列表的部分功能函數(shù).

  1. static PySequenceMethods list_as_sequence = { 
  2.  
  3.     (lenfunc)list_length,                       /* sq_length */ 
  4.  
  5.     (binaryfunc)list_concat,                    /* sq_concat */ 
  6.  
  7.     (ssizeargfunc)list_repeat,                  /* sq_repeat */ 
  8.  
  9.     (ssizeargfunc)list_item,                    /* sq_item */ 
  10.  
  11.     (ssizessizeargfunc)list_slice,              /* sq_slice */ 
  12.  
  13.     (ssizeobjargproc)list_ass_item,             /* sq_ass_item */ 
  14.  
  15.     (ssizessizeobjargproc)list_ass_slice,       /* sq_ass_slice */ 
  16.  
  17.     (objobjproc)list_contains,                  /* sq_contains */ 
  18.  
  19.     (binaryfunc)list_inplace_concat,            /* sq_inplace_concat */ 
  20.  
  21.     (ssizeargfunc)list_inplace_repeat,          /* sq_inplace_repeat */ 
  22.  
  23. }; 

 

接下來(lái)就是一個(gè)判斷, 判斷咱們這個(gè)l對(duì)象是否有Py_TPFLAGS_HAVE_INPLACEOPS這個(gè)特性, 很明顯是有的, 所以就調(diào)用上步取到的結(jié)構(gòu)體中的sq_inplace_concat函數(shù), 那接下來(lái)呢? 肯定就是看看這個(gè)函數(shù)是干嘛的:

  1. list_inplace_concat(PyListObject *self, PyObject *other) 
  2.  
  3.  
  4.     PyObject *result; 
  5.  
  6.   
  7.  
  8.     result = listextend(self, other);    # 關(guān)鍵所在 
  9.  
  10.     if (result == NULL
  11.  
  12.         return result; 
  13.  
  14.     Py_DECREF(result); 
  15.  
  16.     Py_INCREF(self); 
  17.  
  18.     return (PyObject *)self; 
  19.  

 

終于找到關(guān)鍵了, 原來(lái)最后就是調(diào)用這個(gè)listextend函數(shù), 這個(gè)和我們python層面的列表的extend方法很類(lèi)似, 在這不細(xì)講了!

把PyNumber_InPlaceAdd的執(zhí)行調(diào)用過(guò)程, 簡(jiǎn)單整理下來(lái)就是:

  1. INPLACE_ADD(字節(jié)碼) 
  2.  
  3.     -> PyNumber_InPlaceAdd 
  4.  
  5.         -> 判斷是否數(shù)字: 如果是, 直接返回兩數(shù)相加 
  6.  
  7.         -> 判斷是否字符串: 如果是, 直接返回`string_concatenate`的結(jié)果 
  8.  
  9.         -> 都不是: 
  10.  
  11.             -> binary_iop1 (判斷是否數(shù)字, 如果是則按照數(shù)字處理, 否則返回Py_NotImplemented) 
  12.  
  13.                 -> binary_iop (判斷是否數(shù)字, 如果是則按照數(shù)字處理, 否則返回Py_NotImplemented) 
  14.  
  15.             -> 返回的結(jié)果是否 Py_NotImplemented: 
  16.  
  17.                 -> 是: 
  18.  
  19.                     -> 對(duì)象是否有Py_TPFLAGS_HAVE_INPLACEOPS: 
  20.  
  21.                         -> 是: 調(diào)用對(duì)象的: sq_inplace_concat 
  22.  
  23.                         -> 否: 調(diào)用對(duì)象的: sq_concat 
  24.  
  25.                 -> 否: 報(bào)錯(cuò) 

 

所以在上面的結(jié)果, 第二種代碼: l += [3,4,5], 我們看到的id值并沒(méi)有改變, 就是因?yàn)?=通過(guò)sq_inplace_concat調(diào)用了列表的listextend函數(shù), 然后導(dǎo)致新列表以追加的方式去處理.

結(jié)論

現(xiàn)在我們大概明白了+=實(shí)際上是干嘛了: 它應(yīng)該能算是一個(gè)加強(qiáng)版的+, 因?yàn)樗?多了一個(gè)寫(xiě)回本身的功能.不過(guò)是否能夠?qū)懟乇旧? 還是得看對(duì)象自身是否支持, 也就是說(shuō)是否具備Py_NotImplemented標(biāo)識(shí), 是否支持sq_inplace_concat, 如果具備, 才能實(shí)現(xiàn), 否則, 也就是和 + 效果一樣而已. 

責(zé)任編輯:龐桂玉 來(lái)源: Python開(kāi)發(fā)者
相關(guān)推薦

2017-06-16 20:30:54

Python源碼理解

2014-12-25 10:31:33

微信朋友圈挑戰(zhàn)

2019-08-28 12:31:31

戴爾

2013-01-16 10:07:30

加密解密破解Android軟件

2015-04-01 09:09:12

2021-09-07 06:40:26

狀態(tài)機(jī)識(shí)別地址

2024-10-15 09:34:57

2011-04-13 12:46:38

IDF2011凌動(dòng)小尺寸

2020-09-02 07:03:04

虛擬機(jī)HotSpotJava

2020-06-17 15:25:34

Linux 系統(tǒng) 數(shù)據(jù)

2024-08-30 08:50:00

2022-06-01 12:00:54

HTTP狀態(tài)碼服務(wù)端

2017-02-09 15:14:38

物聯(lián)網(wǎng)工信部網(wǎng)號(hào)

2009-12-23 16:15:24

ADO.NET Ent

2012-06-27 11:13:04

x

2018-09-20 16:10:48

CookiesSession前端

2016-06-07 10:28:07

大數(shù)據(jù)機(jī)器學(xué)習(xí)LSTM

2019-09-03 15:43:21

CIOIT經(jīng)理信息化建設(shè)

2009-03-17 18:09:57

虛擬化Vmwareesx

2021-05-10 14:49:21

分析指標(biāo)下跌
點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號(hào)