提出问题

先看以下代码

x = 1
y = 2

tmp = x
x = y
y = tmp

print(x, y)

代码的输出结果是:2 1

再看以下代码:

x = 1
y = 2

x, y = y, x
print(x, y)

代码的输出结果是:2 1

x, y = y, x这段代码背后的含义就是解构和封装

<!--more-->

Python封装

In [1]: t = 1, 2

In [2]: t
Out[2]: (1, 2)

In [3]: type(t)
Out[3]: tuple            # 定义元组是可以省略小括号的

In [4]: t1 = (1, 2)

In [5]: t2 = 1, 2
                        # t1和t2等效
In [6]: t1
Out[6]: (1, 2)

In [7]: t2
Out[7]: (1, 2)

所以封装出来的结果一定是元组

x, y = y, x这段代码的右侧就会封装成(y, x)

Python解构

基本解构

In [8]: lst = [1, 2]

In [9]: first, second = lst

In [10]: print(first, second)
1 2

按照元素顺序,把线性结构lst的元素赋给变量first,second

加星号解构

In [11]: lst = list(range(5))

In [12]: head, *tail = lst

In [13]: head
Out[13]: 0

In [14]: tail
Out[14]: [1, 2, 3, 4]

In [15]: *lst2 = lst    # 左边必须有一个加星号的变量
  File "<ipython-input-15-98211a44ccfb>", line 1
    *lst2 = lst
               ^
SyntaxError: starred assignment target must be in a list or tuple


In [16]: *head, tail = lst

In [17]: head
Out[17]: [0, 1, 2, 3]

In [18]: lst
Out[18]: [0, 1, 2, 3, 4]

In [19]: tail
Out[19]: 4

In [20]: head, *m1, *m2, tail = lst        # 星号不能有多个只能有一个
  File "<ipython-input-20-1fc1a52caa8e>", line 1
    head, *m1, *m2, tail = lst
                              ^
SyntaxError: two starred expressions in assignment


In [21]: v1, v2, v3, v4, v5, v6, v7 = lst    # 左边变量数不能超过右边元素数
---------------------------------------------------------------------------
ValueError                                Traceback (most recent call last)
<ipython-input-21-9366cfb498a1> in <module>()
----> 1 v1, v2, v3, v4, v5, v6, v7 = lst

ValueError: not enough values to unpack (expected 7, got 5)

In [22]: v1, v2 = lst                        #左边变量数不能少于右边元素数
---------------------------------------------------------------------------
ValueError                                Traceback (most recent call last)
<ipython-input-22-d7b0a4e7871e> in <module>()
----> 1 v1, v2 = lst

ValueError: too many values to unpack (expected 2)

总结为以下规律:

  • 元素按照顺序赋值给变量
  • 变量和元素必须匹配
  • 加星号变量,可以接受任意个数的元素
  • 加星号的变量不能单独出现

多层次解构

解构是支持多层次的

In [23]: lst = [1, (2, 3), 5]

In [24]: _, v, *_ = lst        # v解析成(2, 3)

In [25]: v
Out[25]: (2, 3)

In [26]: _, val = v            # v可以进一步解构

In [27]: val
Out[27]: 3

In [28]: _, (_, val), *_ = lst        # 可以一步一次性解构

In [29]: val
Out[29]: 3

In [30]: _, [*_, val], *_ = lst        # 中间部分解构成列表

In [31]: val
Out[31]: 3

In [32]: _, _, val, *_ = lst        # (2, 3)解析成第二个_

In [33]: val
Out[33]: 5

Python下划线的使用

使用单个下划线 _ 表示丢弃该变量,这是Python的一个惯例。单个下划线也是Python合法的标识符, 但是如果不是要丢弃一个变量,通常不要用单个下划线表示一个有意义的变量。可以理解为约定俗成。

解构与封装的使用

非常复杂的数据结构,多层嵌套的线性结构的时候,可以用解构快速提取其中的值,非常的便利

比如以下的使用方法

In [1]: key, _, value = 'I love Python'.partition(' love ')

In [2]: key
Out[2]: 'I'

In [3]: value
Out[3]: 'Python'

记得帮我点赞哦!

精心整理了计算机各个方向的从入门、进阶、实战的视频课程和电子书,按照目录合理分类,总能找到你需要的学习资料,还在等什么?快去关注下载吧!!!

resource-introduce

念念不忘,必有回响,小伙伴们帮我点个赞吧,非常感谢。

我是职场亮哥,YY高级软件工程师、四年工作经验,拒绝咸鱼争当龙头的斜杠程序员。

听我说,进步多,程序人生一把梭

如果有幸能帮到你,请帮我点个【赞】,给个关注,如果能顺带评论给个鼓励,将不胜感激。

职场亮哥文章列表:更多文章

wechat-platform-guide-attention

本人所有文章、回答都与版权保护平台有合作,著作权归职场亮哥所有,未经授权,转载必究!


suncle
59 声望4 粉丝

我是职场亮哥,YY高级工程师、4年工作经验,拒绝咸鱼争当龙头的斜杠程序员


« 上一篇
Python习题集
下一篇 »
Python函数详解