与えられた:
_x = ['w', 'e', 's', 's', 's', 'z','z', 's']
_
s
が出現するたびに、次のインデックスに表示されます。
1位:2
2番目:3
3番目:4
4日:7
x.index('s')
を実行すると、最初のインデックスが取得されます。
4番目のs
のインデックスを取得するにはどうすればよいですか?
オカレンスごとにインデックスを保存したくない場合、または任意の反復可能オブジェクトを操作したい場合は、次のようになります。
from itertools import islice
def nth_index(iterable, value, n):
matches = (idx for idx, val in enumerate(iterable) if val == value)
return next(islice(matches, n-1, n), None)
x = [ 'w', 'e', 's', 's', 's', 'z','z', 's']
idx = nth_index(x, 's', 4)
# 7
None
にはデフォルト値のnext
があることに注意してください。これを別のものに変更するか、削除してStopIteration
をキャッチし、別のより適切な例外として発生させることができます(たとえば、ValueError
は、list.index
の動作とより密接に関連します)。
アイテムのインデックスを取得するには:
return [index for index, char in enumerate(x) if char == 's']
キャラクター自体を取得するために:
return [char for index, char in enumerate(x) if char == 's']
または、文字とインデックスのペアのタプルを取得するには:(より簡単な解決策を指摘してくれたfalsetruに感謝します)
pairs = [(index, char) for index, char in enumerate(x) if char == 's']
def find_nth_character(str1, substr, n):
"""find the index of the nth substr in string str1"""
k = 0
for index, c in enumerate(str1):
#print index, c, n # test
if c == substr:
k += 1
if k == n:
return index
str1 = "B.765.A87_43.Left.9878.xx8"
substr = '.'
occurance = 4
print "%s #%d at index %d" % (substr, occurance, find_nth_character(str1, substr, occurance))
組み込みのリストクラスの機能を拡張するだけです。それを継承することによって。
In [64]: class List(list):
: def __init__(self, *val):
: self.extend(list(val))
:
:
: def findidx(self, val, n=None):
: '''return the occurances of an object in a list'''
:
: if n == None:
: return [i for i, v in enumerate(self) if v == val]
:
: return [i for i, v in enumerate(self) if v == val][n]
このクラスを使用するには2つの方法があります。理解するには、次の例を参照してください。
In [65]: c = List(4, 5, 6, 7, 2, 5, 4 ,4) # enter the elements of the list as a argument
In [69]: c.findidx(4, 0) # search 4's 0th(1) occurance
Out[69]: 0
In [72]: c.findidx(4, 1) # find 4's 1st(2) occurance
Out[72]: 6
または
In [66]: c.findidx(4) # find all occurances of 4
Out[66]: [0, 6, 7]
In [67]: c.findidx(4)[0] # first occurance
Out[67]: 0
In [67]: c.findidx(4)[2] # third occurance
Out[67]: 7
In [69]: c[0]# for verification
Out[69]: 4
In [70]: c[7]
Out[70]: 4
`
_itertools.count
_とジェネレータ式を使用したよりPythonicなアプローチを次に示します。
_In [24]: def get_nth_index(lst, item, n):
...: c = count()
...: return next(i for i, j in enumerate(x) if j=='s' and next(c) == n-1)
_
デモ:
_In [25]: get_nth_index(x, 's', 2)
Out[25]: 3
In [26]: get_nth_index(x, 's', 3)
Out[26]: 4
In [27]: get_nth_index(x, 's', 4)
Out[27]: 7
In [28]: get_nth_index(x, 's', 5)
---------------------------------------------------------------------------
StopIteration Traceback (most recent call last)
<ipython-input-28-fc4e5e8c31ef> in <module>()
----> 1 get_nth_index(x, 's', 5)
<ipython-input-24-5394f79b3c30> in get_nth_index(lst, item, n)
1 def get_nth_index(lst, item, n):
2 c = count()
----> 3 return next(i for i, j in enumerate(x) if j=='s' and next(c) == n-1)
StopIteration:
In [29]:
_
ご覧のとおり、一致するものが見つからない場合はStopIteration
例外が発生します。 next()
関数にデフォルトの引数を渡して、例外を発生させる代わりにデフォルト値を返すこともできます。