Явно выберите элементы из списка или кортежа Python - PullRequest
100 голосов
/ 09 июля 2011

У меня есть следующий список Python (также может быть кортежем):

myList = ['foo', 'bar', 'baz', 'quux']

Я могу сказать

>>> myList[0:3]
['foo', 'bar', 'baz']
>>> myList[::2]
['foo', 'baz']
>>> myList[1::2]
['bar', 'quux']

Как явно выбрать элементы, индексы которых не имеют конкретныхузоры?Например, я хочу выбрать [0,2,3].Или из очень большого списка из 1000 предметов, я хочу выбрать [87, 342, 217, 998, 500].Есть ли какой-то синтаксис Python, который делает это?Нечто похожее:

>>> myBigList[87, 342, 217, 998, 500]

Ответы [ 8 ]

125 голосов
/ 09 июля 2011
list( myBigList[i] for i in [87, 342, 217, 998, 500] )

Я сравнил ответы с python 2.5.2:

  • 19,7 usec: [ myBigList[i] for i in [87, 342, 217, 998, 500] ]

  • 20,6 usec: map(myBigList.__getitem__, (87, 342, 217, 998, 500))

  • 22,7 usec: itemgetter(87, 342, 217, 998, 500)(myBigList)

  • 24,6 usec: list( myBigList[i] for i in [87, 342, 217, 998, 500] )

Обратите внимание, что в Python 3 1-й был изменен, чтобы быть таким же, как 4-й.


Другой вариант - начать с numpy.array, который позволяет индексировать по списку, или numpy.array:

>>> import numpy
>>> myBigList = numpy.array(range(1000))
>>> myBigList[(87, 342, 217, 998, 500)]
Traceback (most recent call last):
  File "<stdin>", line 1, in <module>
IndexError: invalid index
>>> myBigList[[87, 342, 217, 998, 500]]
array([ 87, 342, 217, 998, 500])
>>> myBigList[numpy.array([87, 342, 217, 998, 500])]
array([ 87, 342, 217, 998, 500])

tuple работает не так, как срезы.

36 голосов
/ 09 июля 2011

Что по этому поводу:

from operator import itemgetter
itemgetter(0,2,3)(myList)
('foo', 'baz', 'quux')
9 голосов
/ 09 июля 2011

Он не является встроенным, но вы можете создать подкласс списка, который принимает кортежи в качестве «индексов», если вы хотите:

class MyList(list):

    def __getitem__(self, index):
        if isinstance(index, tuple):
            return [self[i] for i in index]
        return super(MyList, self).__getitem__(index)


seq = MyList("foo bar baaz quux mumble".split())
print seq[0]
print seq[2,4]
print seq[1::2]

печать

foo
['baaz', 'mumble']
['bar', 'quux']
6 голосов
/ 09 июля 2011
>>> map(myList.__getitem__, (2,2,1,3))
('baz', 'baz', 'bar', 'quux')

Вы также можете создать свой собственный класс List, который поддерживает кортежи в качестве аргументов для __getitem__, если вы хотите иметь возможность myList[(2,2,1,3)].

6 голосов
/ 09 июля 2011

Может быть, список в порядке:

L = ['a', 'b', 'c', 'd', 'e', 'f']
print [ L[index] for index in [1,3,5] ]

Производит:

['b', 'd', 'f']

Это то, что вы ищете?

1 голос
/ 18 ноября 2017

Другое возможное решение:

sek=[]
L=[1,2,3,4,5,6,7,8,9,0]
for i in [2, 4, 7, 0, 3]:
   a=[L[i]]
   sek=sek+a
print (sek)
1 голос
/ 01 ноября 2016

Я просто хочу отметить, что даже синтаксис itemgetter выглядит очень аккуратно, но он довольно медленный, когда выполняет большой список.срез занял 0,6225321444745759

0 голосов
/ 16 июля 2018

как часто, когда у вас есть логический массив Numpy, например mask

[mylist[i] for i in np.arange(len(mask), dtype=int)[mask]]

Лямбда, которая работает для любой последовательности или np.array:

subseq = lambda myseq, mask : [myseq[i] for i in np.arange(len(mask), dtype=int)[mask]]

newseq = subseq(myseq, mask)

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...