python 判断数字连续_关于python:检测列表中的连续整数
本問題已經有最佳答案,請猛點這里訪問。
我有一個包含這樣的數據的列表:
[1, 2, 3, 4, 7, 8, 10, 11, 12, 13, 14]
我想打印出連續整數的范圍:
1-4, 7-8, 10-14
是否有內置/快速/高效的方法?
請參閱stackoverflow.com/questions/2154249/,它將您指向docs.python.org/library/itertools.html#examples
家庭作業? 您向我們展示您嘗試過的方法,并看看我們是否可以做得更好。
謝謝,我應該在問之前發現:/
沒問題,并不是那么容易找到-我只是偶然記得看到它。 您的問題并非完全相同,因為您期望的輸出有些不同。
從文檔:
>>> from itertools import groupby
>>> from operator import itemgetter
>>> data = [ 1, 4,5,6, 10, 15,16,17,18, 22, 25,26,27,28]
>>> for k, g in groupby(enumerate(data), lambda (i, x): i-x):
... ? ? print map(itemgetter(1), g)
...
[1]
[4, 5, 6]
[10]
[15, 16, 17, 18]
[22]
[25, 26, 27, 28]
您可以很容易地調整它,以獲取一組打印范圍。
不要忘記import itertools。此外,這僅適用于Python 2.4及更高版本。
實際上,至少在Python 2.6中,您將需要from itertools import *和from operator import *(或同等功能)。
不要使用星級進口!切勿使用星級進口!使用from itertools import groupby和from operator import itemgetter代替。
將lambda更改為lambda ix : ix[0] - ix[1],它可以在Python 3和Python 2中使用(嗯,不計算print語句)。
我打算對這個答案表示贊同,因為它很聰明。不幸的是,如果我沒有解釋代碼的作用/工作原理,就太聰明了。
如果您的輸入列表中有重復項(例如[0,1,1,2]),則此方法將不會對整個序列進行聚類;您需要先將list(set(my_original_list))調用到列表中,然后才能將其提供給此函數以獲取條紋0,1,2
對于所有嘗試使用Python 3的代碼的人,請閱讀@Kevin的注釋。另外,由于您實際上需要使用list(),因此打印語句將不起作用,如您在此處看到的stackoverflow.com/questions/7731213/基本上,您應該在Python 3中使用print(list(map(itemgetter(1), g)))
@ dominic-rodger,如何從groupby()傳遞索引,以便獲得值列表,但又獲得對應索引的另一個列表?我感覺這是可能的,因為您正在枚舉(),但無法實現。
這將完全按照您指定的方式打印:
>>> nums = [1, 2, 3, 4, 7, 8, 10, 11, 12, 13, 14]
>>> ranges = sum((list(t) for t in zip(nums, nums[1:]) if t[0]+1 != t[1]), [])
>>> iranges = iter(nums[0:1] + ranges + nums[-1:])
>>> print ', '.join([str(n) + '-' + str(next(iranges)) for n in iranges])
1-4, 7-8, 10-14
如果列表具有單個數字范圍,則它們將顯示為n-n:
>>> nums = [1, 2, 3, 4, 5, 7, 8, 9, 12, 15, 16, 17, 18]
>>> ranges = sum((list(t) for t in zip(nums, nums[1:]) if t[0]+1 != t[1]), [])
>>> iranges = iter(nums[0:1] + ranges + nums[-1:])
>>> print ', '.join([str(n) + '-' + str(next(iranges)) for n in iranges])
1-5, 7-9, 12-12, 15-18
這需要更多的選票。相當優雅!
一個簡短的解決方案,無需額外導入即可使用。它接受任何迭代,對未排序的輸入進行排序,并刪除重復項:
def ranges(nums):
nums = sorted(set(nums))
gaps = [[s, e] for s, e in zip(nums, nums[1:]) if s+1 < e]
edges = iter(nums[:1] + sum(gaps, []) + nums[-1:])
return list(zip(edges, edges))
例:
>>> ranges([2, 3, 4, 7, 8, 9, 15])
[(2, 4), (7, 9), (15, 15)]
>>> ranges([-1, 0, 1, 2, 3, 12, 13, 15, 100])
[(-1, 3), (12, 13), (15, 15), (100, 100)]
>>> ranges(range(100))
[(0, 99)]
>>> ranges([0])
[(0, 0)]
>>> ranges([])
[]
這與@dansalmo的解決方案相同,我發現了驚人的解決方案,盡管有點難以閱讀和應用(因為它沒有作為函數給出)。
注意,它可以很容易地修改為吐出"傳統的"開放范圍[start, end),例如。更改return語句:
return [(s, e+1) for s, e in zip(edges, edges)]
sehr nett @coldfix
內置:據我所知,沒有。
您必須遍歷數組。首先將第一個值放在變量中并打印出來,然后只要不斷按下下一個數字,除了記住另一個變量中的最后一個數字,什么都不會做。如果下一個號碼不符合要求,請檢查記住的最后一個號碼與第一個號碼。如果相同,則什么也不做。如果不同,則打印"-"和最后一個數字。然后將當前值放在第一個變量中,然后重新開始。
在數組的末尾,您運行相同的例程,就好像碰到了一個不符合要求的數字一樣。
我當然可以編寫代碼,但是我不想破壞您的作業:-)
使用設置操作,可以執行以下算法
def get_consecutive_integer_series(integer_list):
integer_list = sorted(integer_list)
start_item = integer_list[0]
end_item = integer_list[-1]
a = set(integer_list) ?# Set a
b = range(start_item, end_item+1)
# Pick items that are not in range.
c = set(b) - a ?# Set operation b-a
li = []
start = 0
for i in sorted(c):
end = b.index(i) ?# Get end point of the list slicing
li.append(b[start:end]) ?# Slice list using values
start = end + 1 ?# Increment the start point for next slicing
li.append(b[start:]) ?# Add the last series
for sliced_list in li:
if not sliced_list:
# list is empty
continue
if len(sliced_list) == 1:
# If only one item found in list
yield sliced_list[0]
else:
yield"{0}-{1}".format(sliced_list[0], sliced_list[-1])
a = [1, 2, 3, 6, 7, 8, 4, 14, 15, 21]
for series in get_consecutive_integer_series(a):
print series
上面的列表" a"的輸出
1-4
6-8
14-15
21
我有一個類似的問題,正在將以下內容用于排序列表。它輸出具有字典中列出的值范圍的字典。鍵將連續數字的每次運行分開,也按順序排列數字之間的非連續項目的運行總數。
您的列表給了我{0: [1, 4], 1: [7, 8], 2: [10, 14]}的輸出
def series_dictf(index_list):
from collections import defaultdict
series_dict = defaultdict(list)
sequence_dict = dict()
list_len = len(index_list)
series_interrupts = 0
for i in range(list_len):
if i == (list_len - 1):
break
position_a = index_list[i]
position_b = index_list[i + 1]
if position_b == (position_a + 1):
sequence_dict[position_a] = (series_interrupts)
sequence_dict[position_b] = (series_interrupts)
if position_b != (position_a + 1):
series_interrupts += 1
for position, series in sequence_dict.items():
series_dict[series].append(position)
for series, position in series_dict.items():
series_dict[series] = [position[0], position[-1]]
return series_dict
您可以使用具有稱為Counter的類的集合庫。如果嘗試輪詢任何可迭代的不同元素的數量,則計數器可以派上用場
from collections import Counter
data = [ 1, 4,5,6, 10, 15,16,17,18, 22, 25,26,27,28]
cnt=Counter(data)
print(cnt)
此輸出看起來像
Counter({1: 1, 4: 1, 5: 1, 6: 1, 10: 1, 15: 1, 16: 1, 17: 1, 18: 1, 22: 1, 25: 1, 26: 1, 27: 1, 28: 1})
就像其他字典一樣,可以查詢鍵值
多數民眾贊成在根本沒有回答問題,您只計算發生的次數而OP要求ranges
這是不使用任何模塊的另一種基本解決方案,非常適合面試,通常在他們不使用任何模塊的情況下進行面試:
#!/usr/bin/python
def split_list(n):
"""will return the list index"""
return [(x+1) for x,y in zip(n, n[1:]) if y-x != 1]
def get_sub_list(my_list):
"""will split the list base on the index"""
my_index = split_list(my_list)
output = list()
prev = 0
for index in my_index:
new_list = [ x for x in my_list[prev:] if x < index]
output.append(new_list)
prev += len(new_list)
output.append([ x for x in my_list[prev:]])
return output
my_list = [1, 3, 4, 7, 8, 10, 11, 13, 14]
print get_sub_list(my_list)
輸出:
[[1], [3, 4], [7, 8], [10, 11], [13, 14]]
總結
以上是生活随笔為你收集整理的python 判断数字连续_关于python:检测列表中的连续整数的全部內容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: JS基础之undefined与null的
- 下一篇: java适配器模式_适配器模式的设计,你