import operator
ranges = {
'4' : 'a',
'70' : 'b',
'700': 'c',
'701': 'd',
'85' : 'e',
'87' : 'a',
}
def id_for_value(value):
possible = '*'
for idvalue, id in sorted(ranges.iteritems()):
if value.startswith(idvalue):
possible = id
elif idvalue > value:
break
return possible
知道某个值的id就足够了。测试:
assert id_for_value('10') == '*'
assert id_for_value('499') == 'a'
assert id_for_value('703') == 'b'
assert id_for_value('7007') == 'c'
assert id_for_value('7017') == 'd'
assert id_for_value('76') == id_for_value('83') == '*'
assert id_for_value('857') == 'e'
assert id_for_value('8716') == 'a'
如果你真的想要范围,你可以使用 itertools.groupby 来计算它:
def firstlast(iterator):
""" Returns the first and last value of an iterator"""
first = last = iterator.next()
for value in iterator:
last = value
return first, last
maxlen = max(len(x) for x in ranges) + 1
test_range = ('%0*d' % (maxlen, i) for i in xrange(10 ** maxlen))
result = dict((firstlast(gr), id)
for id, gr in itertools.groupby(test_range, key=id_for_value))
给出:
{('0000', '3999'): '*',
('4000', '4999'): 'a',
('5000', '6999'): '*',
('7000', '7009'): 'c',
('7010', '7019'): 'd',
('7020', '7099'): 'b',
('7100', '8499'): '*',
('8500', '8599'): 'e',
('8600', '8699'): '*',
('8700', '8799'): 'a',
('8800', '9999'): '*'}