### 问题:
如果一个可迭代对象的元素个数超过变量个数时,会抛出一个 ValueError。
那么怎样才能从这个可迭代对象中解压出 N个元素出来?
*****
### 解决方案:
Python 的星号表达式可以用来解决这个问题,比如,你在学习一门课程,在学期末的时候
你想统计下家庭作业的平级成绩,但是排除掉第一个和最后一个分数。如果只有四个分数,
你可能就直接去简单的手动赋值,但如果有 24 个呢?这时候星号表达式就派上用场了。
*****
例子:
```python
grades = [60, 40, 70, 80, 65, 49, 12, 34, 99, 75, 46, 71, 68, 22, 81, 98, 75, 65, 15, 75]
def avg(middle):
print(middle)
def drop_fist_last(grades):
print(sorted(grades))
first, *middle, last = sorted(grades)
return avg(middle)
drop_fist_last(grades)
```
输出:
```
[12, 15, 22, 34, 40, 46, 49, 60, 65, 65, 68, 70, 71, 75, 75, 75, 80, 81, 98, 99]
[15, 22, 34, 40, 46, 49, 60, 65, 65, 68, 70, 71, 75, 75, 75, 80, 81, 98]
```
### 另外一种情况,假设你现在有一些用户的记录列表,每条记录包含一个名字,邮件,接着就是不确定数量的电话号码。你可以像下面这样分解这些记录
```Python
record = ('Dave', 'dave@example.com', '773-555-1212', '847-555-1212')
name, email, *phone_numbers = record
print(name) # Dave
print(email) # dave@example.com
print(phone_numbers) # ['773-555-1212', '847-555-1212']
```
### 值得注意的是上面解压出来的 `phone_numbers` 变量永远是列表类型,不管解压的电话号码数量是多少(包括0个)。所以,任何使用到 `phone_numbers` 变量的代码就不需要做多余的类型检查去确认它是否是列表类型了。
**星号表达式也能用在列表的开始部分**
```Python
*trailing, current = [10, 8, 7, 1, 9, 5, 10, 3]
print(trailing) # [10, 8, 7, 1, 9, 5, 10]
print(current) # 3
```
### 扩展的迭代解压语法是专门为解压不确定个数或任意个数元素的可迭代对象而设计的。通常,这些可迭代对象的元素结构有确定的规则(比如第一个元素后面都是电话号码),星号表达式让开发人员可以很容易的利用这些规则来解压出元素来。而不是通过一些比较复杂的手段去获取这些关联的元素值。
** 星号表达式在迭代元素为可变长元组的序列时是很有用的。比如,下面是一个带有标签的元组序列**
```Python
records = [
('foo', 1, 2),
('bar', 'hello'),
('foo', 3, 4)
]
def do_foo(x,y):
print('foo',x,y)
def do_bar(s):
print('bar',s)
for tag, *args in records:
if tag == 'foo':
do_foo(*args)
else:
do_bar(*args)
```
输出:
```
foo 1 2
bar hello
foo 3 4
```
** 星号解压语法在字符串操作的时候也会很有用。比如字符串的分割。**
```Python
line = 'nobody:*:-2:-2:Unprivileged User:/var/empty:/usr/bin/false'
uname, *fields, homedir, sh = line.split(':')
print(uname) # nobody
print(fields) # ['*', '-2', '-2', 'Unprivileged User']
print(homedir) # /var/empty
print(sh) # /usr/bin/false
```
*****
**有时候,你想解压一些元素后丢弃它们,你不能简单就使用 * ,但是你可以使用一个普通的废弃名称,比如 _ 或者 ign (ignore)**
```Python
record = ('ACME', 50, 123.45, (12, 18, 2018))
name, *_, (*_, year) = record
print(name) # ACME
print(year) # 2018
```
*****
**在很多函数式语言中,星号解压语法跟列表处理有很多相似之处。比如,如果你有一个列表,你可以很容易的将它分割成前后两部分**
```Python
items = [1, 10, 7, 4, 5, 9]
head, *tail, last = items
print(head) # 1
print(last) # 9
```
*****
**还能用这种分割语法去巧妙的实现递归算法。**
```Python
items = [1, 10, 7, 4, 5, 9]
def sums(items):
head, *tail = items
return head + sums(tail) if tail else head
```
原文:
> [http://python3-cookbook.readthedocs.io/zh\_CN/latest/c01/p02\_unpack\_elements\_from\_iterables.html](http://python3-cookbook.readthedocs.io/zh_CN/latest/c01/p02_unpack_elements_from_iterables.html)