python 《CookBook》 从饿死到撑死

第一章 数据结构和算法(1)

2019-01-08  本文已影响0人  楠木cral

1、将序列分解为单独的变量

1.1 问题描述
将一个包含n个元素的元祖或者序列分解为n个独立的变量
1.2 解决方案
任何可迭代的对象(不仅仅是元组或者列表),都可以通过赋值操作来分解成单独的变量,但是变量的总数量(变量个数)和结构(后面有例子说明这个结构)要与这个对象吻合

tup = (1, 2, 3)
a, b, c = tup
print(a, b, c)    # 1 2 3
data = ['wyl', 520, 13.14, (2019, 1, 8)]
name, action, time, date = data
print(name, action, time, date)      # wyl 520 13.14 (2019, 1, 8)

下面想把年月日分开,那么也要和data里面的一样写成元祖结构 (year, month, day)才能赋值分解,不然会报错

data = ['wyl', 520, 13.14, (2019, 1, 8)]
name, action, time, (year, month, day) = data
print(name, action, time, year, month, day)     # wyl 520 13.14 2019 1 8

前面说了,只要是可以迭代的对象都可以执行分解操作(字符串、文件、生成器等)

s = 'hello'
a, b, c, d, e = s
print(a, b, c, d, e)   # h e l l o

注意:如果想在这个迭代对象中某些值不要,那么还是得用变量名把它取出来,但是要确保这个变量名以前没用过

2.从任意长度的可迭代的对象中分解元素

2.1 问题描述
需要从某个可迭代对象中分解n个元素,但是这个对象长度可能超过n,从而导致“ 分解值过多(too many values to unpank)”的异常
2.2 解决方案
利用python中的“*表达式”,可以自己理解为‘不定长参数’那样的

record = ('Mary', '123456789@qq.com', '2426529', '5201314')
name, email, *tel = record
print(name)   # Mary
print(email)  # 123456789@qq.com
print(*tel)   # 2426529 5201314 

这个可以简单地理解为,先按照位置匹配不是‘*表达式’的,剩下的都匹配“ *表达式 ”,改变 *tel 的位置看看:

record = ('Mary', '2426529', '5201314', '123456789@qq.com')
name, *tel, email = record
print(name)   # Mary
print(*tel)   # 2426529 5201314
print(email)  # 123456789@qq.com
注意:“ *表达式 ”匹配出来的是一个列表类型

对于分解未知或者任意长度的可迭代对象,这个 *表达式 就显得尤为优秀,这里举一个迭代变长的元祖序列:

record =[
    ('foo', 1, 3),
    ('bar', 'lalala'),
    ('foo', 5, 2),
]
def do_foo(x, y):
    print('foo', x, y)
def do_bar(s):
    print('bar', s)
for tag, *args in record:
    if tag == 'foo':
        do_foo(*args)
    elif tag == 'bar':
        do_bar(*args)
# foo 1 3
# bar lalala
# foo 5 2

当和某些特定的字符串处理操作相结合,比如做拆分操作,*式所支持的分解操作也非常有用,例如:

line = 'nobody:*:15:-952:hijhugjkhHJ User:/zfkjn/fsg/use/bin'
uname, *filed, hom, sh = line.split(':')
print(uname)   # nobody
print(*filed)   # * 15 -952   # 这是列表['*', '15', '-952']
print(hom)  # hijhugjkhHJ User  # 这是含空格的字符串'hijhugjkhHJ User'
print(sh)   # /zfkjn/fsg/use/bin

如果你想分解出来丢弃掉某些值,则可以用几个常用的想丢弃的变量名,比如*_或者ign(ignored):

record = ('wyl', 52, 0, 13.14, (7, 7, 2018))
name, *_, (*_, year) = record
print(name)  # wyl
print(year)  # 2018

可以用这个实现假设的递归函数:

items = [1, 10, 7, 8, 9, 6]
def sum(items):
    head, *tail = items
    return head + sum(tail) if tail else head
print(sum(items))  # 41
上一篇 下一篇

猜你喜欢

热点阅读