亚洲综合原千岁中文字幕_国产精品99久久久久久久vr_无码人妻aⅴ一区二区三区浪潮_成人h动漫精品一区二区三

主頁 > 知識庫 > Python源碼解析之List

Python源碼解析之List

熱門標簽:沈陽防封電銷卡品牌 池州外呼調研線路 沈陽人工外呼系統價格 富錦商家地圖標注 沈陽外呼系統呼叫系統 武漢外呼系統平臺 如何申請400電話費用 外呼系統哪些好辦 江西省地圖標注

一、列表結構體

創建列表C語言底層的結構體

lists = []
list.append('name')
list.append('age')
list.append('grade')
typedef struct{
	struct _object *_ob_next;
	struct _object *_ob_prev; 	// python內部將對象放在鏈表進行內存管理
	Py_ssize_t ob_refcnt;		// 引用計數器,就是多少變量用了它
	PyObject **ob_item;			// 指針的指針,存列表的元素
	Py_ssize_t ob_size;			// 已有元素個數
	Py_ssize_t allocated;		// 列表容量,可容納個數
} PyListObject;

c源碼來自 listobject.c

二、創建列表

name_list = [ ]

PyObject *
PyList_New(Py_ssize_t size)
{
    PyListObject *op;
    size_t nbytes;
#ifdef SHOW_ALLOC_COUNT
    static int initialized = 0;
    if (!initialized) {
        Py_AtExit(show_alloc);
        initialized = 1;
    }
#endif
    // 緩存機制
    if (size  0) {
        PyErr_BadInternalCall();
        return NULL;
    }
    /* Check for overflow without an actual overflow,
     *  which can cause compiler to optimise out */
    if ((size_t)size > PY_SIZE_MAX / sizeof(PyObject *))
        return PyErr_NoMemory();
    nbytes = size * sizeof(PyObject *);
    if (numfree) {
        numfree--;
        op = free_list[numfree];
        _Py_NewReference((PyObject *)op);
#ifdef SHOW_ALLOC_COUNT
        count_reuse++;
#endif
    } else {
        op = PyObject_GC_New(PyListObject, PyList_Type);
        if (op == NULL)
            return NULL;Py
#ifdef SHOW_ALLOC_COUNT
        count_alloc++;
#endif
    }

    if (size = 0)
        op->ob_item = NULL;
    else {
        op->ob_item = (PyObject **) PyMem_MALLOC(nbytes);
        if (op->ob_item == NULL) {
            Py_DECREF(op);
            return PyErr_NoMemory();
        }
        memset(op->ob_item, 0, nbytes);
    }
    Py_SIZE(op) = size;  // 元素個數
    op->allocated = size;   // 容量
    _PyObject_GC_TRACK(op); //放到雙向鏈表進行維護
    return (PyObject *) op; //返回列表的指針
}

三、添加元素

list中插入一個元素時,擴容連續的內存地址(容量),在內存創建需要插入的內容p,將地址*p放入list的空間中,所以,PyListObject的ob_item是指針的指針

擴容的曲線一般就是0,4,8,16,24…

// 添加元素
static int
app1(PyListObject *self, PyObject *v)
{
    // 獲取實際元素個數
    Py_ssize_t n = PyList_GET_SIZE(self);

    assert (v != NULL);
    if (n == PY_SSIZE_T_MAX) {
        PyErr_SetString(PyExc_OverflowError,
            "cannot add more objects to list");
        return -1;
    }

    // 計算當前容量和內部元素個數
    // 直接添加元素/擴容添加
    if (list_resize(self, n+1) == -1)
        return -1;
    // 將元素添加到ob_item,v
    Py_INCREF(v);
    PyList_SET_ITEM(self, n, v);
    return 0;
}
  • 擴容
// 擴容機制
 // newsize: 已存在元素個數+1
static int
list_resize(PyListObject *self, Py_ssize_t newsize)
{
    PyObject **items;
    size_t new_allocated;
    Py_ssize_t allocated = self->allocated; // 當前的容量

    // 1,容量大于個數
    // 2,個數大于容量的一半(容量足夠且沒有內存浪費)
    if (allocated >= newsize  newsize >= (allocated >> 1)) {
        assert(self->ob_item != NULL || newsize == 0);
        Py_SIZE(self) = newsize;
        return 0;
    }

    /* 
     * The growth pattern is:  0, 4, 8, 16, 25, 35, 46, 58, 72, 88, ...
     */
     // 擴容機制的算法
    new_allocated = (newsize >> 3) + (newsize  9 ? 3 : 6);

    /* check for integer overflow */
    if (new_allocated > PY_SIZE_MAX - newsize) {
        PyErr_NoMemory();
        return -1;
    } else {
        new_allocated += newsize;
    }

    if (newsize == 0)
        new_allocated = 0;
    // 擴容/縮容(涉及原來元素的遷移)
    items = self->ob_item;
    if (new_allocated = (PY_SIZE_MAX / sizeof(PyObject *)))
        PyMem_RESIZE(items, PyObject *, new_allocated);
    else
        items = NULL;
    if (items == NULL) {
        PyErr_NoMemory();
        return -1;
    }
    // 賦值,更新個數和容量
    self->ob_item = items;
    Py_SIZE(self) = newsize;
    self->allocated = new_allocated;
    return 0;
}

四、移除元素

list.pop()
刪除最后一個元素只需要修改size,不需要清除數據,下次append可以直接覆蓋這個位置
指定索引位置移除后,向前補位

static PyObject *
listpop(PyListObject *self, PyObject *args)
{
    Py_ssize_t i = -1;
    PyObject *v;
    int status;

    if (!PyArg_ParseTuple(args, "|n:pop", i))
        return NULL;

    if (Py_SIZE(self) == 0) {
        /* Special-case most common failure cause */
        PyErr_SetString(PyExc_IndexError, "pop from empty list");
        return NULL;
    }
    if (i  0)
        i += Py_SIZE(self);
    if (i  0 || i >= Py_SIZE(self)) {
        PyErr_SetString(PyExc_IndexError, "pop index out of range");
        return NULL;
    }
    v = self->ob_item[i];
    // 刪除最后一個,僅改變size
    if (i == Py_SIZE(self) - 1) {
        status = list_resize(self, Py_SIZE(self) - 1);
        assert(status >= 0);
        return v; /* and v now owns the reference the list had */
    }
    Py_INCREF(v);
    // 不是最后一個,需要移動數據位置
    status = list_ass_slice(self, i, i+1, (PyObject *)NULL);
    assert(status >= 0);
    /* Use status, so that in a release build compilers don't
     * complain about the unused name.
     */
    (void) status;

    return v;
}

五、清空

list.clear()

static int
list_clear(PyListObject *a)
{
    Py_ssize_t i;
    PyObject **item = a->ob_item;
    if (item != NULL) {
        i = Py_SIZE(a);
        // 各個元素設置為空
        Py_SIZE(a) = 0;
        a->ob_item = NULL;
        a->allocated = 0;
        // 引用計數器-1
        while (--i >= 0) {
            Py_XDECREF(item[i]);
        }
        PyMem_FREE(item);
    }
 
    return 0;
}

六、銷毀

del list

銷毀列表對象的操作
將列表的引用計數-1
引用計數>0,還有應用的話不做操作
引用計數=0,沒人使用

  • 處理列表的元素,將所有引用計數-1(GC回收0計數)
  • ob_item=0,ob_size=0,ob_allocated=0
  • 將列表從雙向鏈表移除,可以銷毀
  • 為了提高效率,Python結束期在內部為free_list緩存80個list,存放無使用的list,再創建的時候直接從緩存中拿來初始化。如果已經存了80個,del 的時候直接在內存中銷毀對象
static void
list_dealloc(PyListObject *op)
{
    Py_ssize_t i;
    // 判斷引用計數是否為0
    PyObject_GC_UnTrack(op);
    Py_TRASHCAN_SAFE_BEGIN(op)
    if (op->ob_item != NULL) {
        i = Py_SIZE(op);
        while (--i >= 0) {
            Py_XDECREF(op->ob_item[i]);
        }
        PyMem_FREE(op->ob_item);
    }
    // free_list沒有80個的話緩存這個list
    if (numfree  PyList_MAXFREELIST  PyList_CheckExact(op))
        free_list[numfree++] = op;
    else
        Py_TYPE(op)->tp_free((PyObject *)op);
    Py_TRASHCAN_SAFE_END(op)
}

就是說創建列表時,實際上不會直接開辟內存,而是先看看free_list

# 兩次list的地址相同
>>> list1=[1,2,3]
>>> id(list1)
69070216L
>>> del list1
>>> list2=[0,0,0]
>>> id(list2)
69303304L
>>> 

到此這篇關于Python源碼解析之List的文章就介紹到這了,更多相關Python List內容請搜索腳本之家以前的文章或繼續瀏覽下面的相關文章希望大家以后多多支持腳本之家!

您可能感興趣的文章:
  • Python源碼學習之PyObject和PyTypeObject
  • python源碼剖析之PyObject詳解
  • python opencv人臉識別考勤系統的完整源碼
  • python 制作手機歸屬地查詢工具(附源碼)
  • python基于tkinter制作無損音樂下載工具(附源碼)
  • Python bsonrpc源碼解讀
  • Python源碼學習之PyType_Type和PyBaseObject_Type詳解
  • Python制作腳本幫女朋友搶購清空購物車

標簽:呂梁 銅川 通遼 潛江 黑龍江 常德 阿里 株洲

巨人網絡通訊聲明:本文標題《Python源碼解析之List》,本文關鍵詞  Python,源碼,解析,之,List,;如發現本文內容存在版權問題,煩請提供相關信息告之我們,我們將及時溝通與處理。本站內容系統采集于網絡,涉及言論、版權與本站無關。
  • 相關文章
  • 下面列出與本文章《Python源碼解析之List》相關的同類信息!
  • 本頁收集關于Python源碼解析之List的相關信息資訊供網民參考!
  • 推薦文章
    999精品影视在线观看| 国产一区二区精品久久| 精品视频在线观看一区二区| 精品视频一区二区| 精品毛片视频| 欧美国产日韩在线| 久久国产一区二区| 国产成人啪精品| 成人免费网站视频ww| 精品国产香蕉伊思人在线又爽又黄| 四虎久久精品国产| 一级女人毛片人一女人| 亚洲 国产精品 日韩| 精品在线视频播放| 国产成+人+综合+亚洲不卡| 午夜激情视频在线观看| 午夜激情视频在线播放| 精品视频一区二区| 久久精品店| 日本伦理网站| 精品视频在线观看一区二区| 精品久久久久久中文| 日日爽天天| 四虎久久影院| 国产原创中文字幕| 九九精品在线| 精品视频在线看 | 欧美18性精品| 91麻豆精品国产综合久久久| 国产伦精品一区二区三区无广告| 久久久久久久男人的天堂| 欧美a免费| 韩国毛片基地| 夜夜操网| 成人高清免费| 欧美大片毛片aaa免费看| 999久久久免费精品国产牛牛| 麻豆网站在线看| 亚洲 国产精品 日韩| 精品视频在线看 | 国产一级生活片| 午夜精品国产自在现线拍| a级毛片免费观看网站| 深夜做爰性大片中文| 亚洲第一色在线| 国产美女在线一区二区三区| 天天色成人网| 亚久久伊人精品青青草原2020| 精品国产一区二区三区精东影业| 久久国产精品只做精品| 日本免费乱理伦片在线观看2018| 欧美激情影院| 日韩专区一区| 日韩在线观看视频免费| 久久精品免视看国产成人2021| 欧美夜夜骑 青草视频在线观看完整版 久久精品99无色码中文字幕 欧美日韩一区二区在线观看视频 欧美中文字幕在线视频 www.99精品 香蕉视频久久 | 精品国产一区二区三区国产馆| 欧美国产日韩一区二区三区| 天天色色色| 九九干| 二级特黄绝大片免费视频大片| 日韩在线观看视频网站| 欧美另类videosbestsex久久| 超级乱淫伦动漫| 精品国产亚洲人成在线| 精品国产一区二区三区精东影业 | 天堂网中文在线| 免费国产在线观看不卡| 国产一区二区精品| 尤物视频网站在线| 一级女性全黄生活片免费| 欧美国产日韩一区二区三区| 国产视频一区在线| 香蕉视频久久| 国产伦精品一区二区三区无广告| a级黄色毛片免费播放视频| 91麻豆精品国产自产在线观看一区| 四虎影视久久久| 精品国产亚一区二区三区| 成人免费观看视频| 中文字幕一区二区三区精彩视频| 日韩综合| 国产视频网站在线观看| 99久久精品国产麻豆| 日韩在线观看免费完整版视频| 日韩av成人| 免费毛片基地| 成人免费观看男女羞羞视频| 九九免费高清在线观看视频| 国产精品自拍在线| 免费一级片在线| 欧美激情伊人| 天天做日日爱| 色综合久久天天综合| 精品国产一区二区三区精东影业 | 一本高清在线| 99色视频| 日本在线www| 日韩av片免费播放| 青草国产在线观看| 日本伦理黄色大片在线观看网站| 国产a视频精品免费观看| 精品国产一区二区三区久久久蜜臀 | 夜夜操网| 精品久久久久久综合网| 午夜在线亚洲| 久草免费在线色站| 欧美大片aaaa一级毛片| 999久久狠狠免费精品| 一 级 黄 中国色 片| 久久久成人影院| 四虎论坛| 久久精品欧美一区二区| 国产成a人片在线观看视频| 久久福利影视| 久久久成人影院| 91麻豆tv| 99热精品一区| 日韩av成人| 香蕉视频亚洲一级| 夜夜操网| 日韩av成人| 国产伦久视频免费观看 视频| 国产高清在线精品一区a| 日韩专区一区| 欧美另类videosbestsex高清| 国产伦精品一区二区三区无广告 | 国产一区二区精品| 成人免费高清视频| 成人在免费观看视频国产| 日本特黄特色aaa大片免费| 九九精品久久| | 日日爽天天| 亚洲天堂在线播放| 超级乱淫伦动漫| 欧美一级视频高清片| 国产视频网站在线观看| 尤物视频网站在线观看| 亚洲 欧美 成人日韩| 日韩在线观看视频网站| 亚洲精品影院| 国产成人啪精品| 国产精品自拍在线观看| 好男人天堂网 久久精品国产这里是免费 国产精品成人一区二区 男人天堂网2021 男人的天堂在线观看 丁香六月综合激情 | 国产网站在线| 久草免费在线观看| 国产视频一区二区在线观看 | 成人高清护士在线播放| 日本在线不卡免费视频一区| 精品国产一区二区三区精东影业 | 国产成人精品综合| 免费一级生活片| 亚洲精品久久久中文字| 黄色免费三级| 韩国三级视频在线观看| 国产一区二区精品尤物| 黄视频网站免费| 黄色福利| 久久久久久久男人的天堂| 亚洲女初尝黑人巨高清在线观看| 九九精品在线| 久久精品成人一区二区三区| 精品国产一区二区三区免费 | | 国产高清在线精品一区二区| 日韩中文字幕在线播放| 久久久成人影院| 成人免费高清视频| 欧美激情一区二区三区视频| 天天色色色| 天堂网中文字幕| 国产成人啪精品| 日韩中文字幕一区二区不卡| 美女免费毛片| 国产美女在线一区二区三区| 国产亚洲免费观看| 欧美1区| 欧美18性精品| 黄视频网站在线免费观看| 久久99中文字幕| 久久99中文字幕久久| 欧美a免费| 国产极品白嫩美女在线观看看| 国产一区二区精品尤物| 香蕉视频久久| 91麻豆国产福利精品| 九九免费精品视频| 999精品视频在线| 日日夜夜婷婷| 91麻豆精品国产片在线观看| 国产一区二区精品久久91| 精品视频免费观看| 成人免费观看男女羞羞视频| 久久国产精品永久免费网站| 免费一级片在线观看| 欧美大片aaaa一级毛片| 精品国产一区二区三区精东影业| 亚洲精品久久久中文字| 日韩av成人|