簡體   English   中英

如何創建超過Python最大大小的列表

[英]How to create a list that exceeds the maximum size in Python

根據在32位系統上Python列表的最大大小為536870912層的元件

有沒有可能的方法來初始化比這更大的列表?

讓我們說:

list1 = [None]*1000000000

由於列表中的每個元素至少占用4個字節,所以列出大的占用大量空間,因此只有一個包含最大允許元素的列表將占用最少2GB的RAM 1 而這甚至沒有考慮64位系統2

  1. 4 * 5.4e+8 = 2.1e+9 GB
  2. 8 * 1.2e+18 = 9.2e+18 EB(是,exabytes)

但是,為了這個問題,讓我們假設你有很多內存。


最簡單的選擇之一是創建自己的對象來保存和處理大量列表。 它基本上將大量列表拆分為較小的列表,然后在需要時相應地訪問它們。 由於你不得不重寫所有方法,因為沒有子類化list真正好處 ,你最好只是繼承object ,然后從那里開始。 這里唯一必不可少的是永遠不要組合或復制子列表(因為它們很大),所以在必要時使用itertools.chain循環遍歷列表。

以下是最簡單的list-methods appendextendget/setitem工作的get/setitem

import sys
from itertools import chain

class largeList(object):
    def __init__(self, mylist=[]):
        self.maxSize = sys.maxsize/4
        self.src = [[]]
        self.extend(mylist)

    def __iter__(self):
        return chain(*self.src)

    def __getitem__(self, idx):
        return self.src[int(idx/self.maxSize)][idx%self.maxSize]

    def __setitem__(self, idx, item):
        self.src[int(idx/self.maxSize)][idx%self.maxSize] = item
        # expand set/getitem to support negative indexing.

    def append(self, item):
        if len(self.src[-1]) < self.maxSize:
            self.src[-1].append(item)
        else:
            self.src.append([item])

    def extend(self, items):
        remainder = self.maxSize - len(self.src[-1])
        self.src[-1].extend(items[:remainder])
        for i in xrange(0, len(items[remainder:]), self.maxSize):
            self.src.append(items[remainder:][i:i+self.maxSize])

    def __len__(self):
        return sum(len(l) for l in self.src)

    def __str__(self):
        size = self.__len__()
        if size >= 8:
            first, last = [], []
            for i, ele in enumerate(self.__iter__()):
                if i < 3:
                    first.append(ele)
                if i >= size - 3:
                    last.append(ele)
            return str(first)[:-1] + ', ..., ' + str(last)[1:]
        return str(list(self.__iter__()))

使用示例( 如果您的可用RAM少於4GB或者您使用的是64位系統, sys.maxsize在嘗試之前更改sys.maxsize ):

#sys.maxsize = 1000

list1 = largeList(xrange(sys.maxsize/4 + 2))
print len(list1)
# 53687093
print list1
#[0, 1, 2, ..., 536870910, 536870911, 536870912]
print list1.src
#[[0, 1, 2 ..., 536870910], [536870911, 536870912]]
list1.extend([42, 43])
print list1
#[0, 1, 2, ..., 536870912, 42, 43]

結果:在列表內部將列表拆分為多個列表,而在使用它們時它們似乎只是一個列表。 通過添加更多方法,可以輕松添加更多list功能。 例如popremoveinsertindex

(...)

    def pop(self, idx):
        listidx = int(idx/self.maxSize)
        itempopped = self.src[listidx].pop(idx%self.maxSize)
        for i in xrange(listidx, len(self.src)-1):
            self.src[i].append(self.src[i+1].pop(0))
        if not self.src[-1]:
            del self.src[-1]
        return itempopped

    def remove(self, item):
        for i, ele in enumerate(self.__iter__()):
            if ele == item:
                self.pop(i)
                break

    def insert(self, idx, item):
        listidx = int(idx/self.maxSize)
        itemtoshift = self.src[listidx].pop(-1)
        self.src[listidx].insert(idx%self.maxSize, item)
        for i in xrange(listidx+1, len(self.src)-1):
            itemremoved = self.src[i].pop(-1)
            self.src[i].insert(0, itemtoshift)
            itemtoshift = itemremoved
        if len(self.src[-1]) < self.maxSize:
            self.src[-1].insert(0, itemtoshift)
        else:
            self.src.append([self.src[-1].pop(-1)])
            self.src[-2].insert(0, itemtoshift)

    def index(self, item):
        for i, ele in enumerate(self.__iter__()):
            if ele == item:
                return i
        return -1

示例使用,續:

#sys.maxsize = 1000

list1 = largeList(xrange(sys.maxsize/4 + 2))
list1.insert(0, 'a')
print list1
#['a', 0, 1, ..., 536870910, 536870911, 536870912]
list1.pop(2)
#1
list1.remove(536870910)
print list1.index('a')
#0
print len(list1)
#536870911
print list1
#['a', 0, 2, ..., 536870909, 536870911, 536870912]
print list.src
#[['a', 0, 2, ..., 536870909, 536870911], [536870912]]

暫無
暫無

聲明:本站的技術帖子網頁,遵循CC BY-SA 4.0協議,如果您需要轉載,請注明本站網址或者原文地址。任何問題請咨詢:yoyou2525@163.com.

 
粵ICP備18138465號  © 2020-2024 STACKOOM.COM