Python列表嵌套常见坑点及解决方案


Posted in Python onSeptember 30, 2020

1.嵌套列表

Python中有一种内置的数据类型叫列表(list),它是一种容器,可以用来承载其他的对象(准确的说是其他对象的引用),列表中的对象可以称为列表的元素,很明显我们可以把列表作为列表中的元素,这就是所谓的嵌套列表。

嵌套列表可以模拟出现实中的表格、矩阵、2D游戏的地图(如植物大战僵尸的花园)、棋盘(如国际象棋、黑白棋)等。

2.识别坑点

在使用嵌套的列表时要小心,否则很可能遭遇非常尴尬的情况,下面是一个小例子。

def main():  names = ['关羽', '张飞', '赵云', '马超', '黄忠']  subjs = ['语文', '数学', '英语']  scores = [[0] * 3] * 5  for row, name in enumerate(names):    print('请输入%s的成绩' % name)    for col, subj in enumerate(subjs):      scores[row][col] = float(input(subj + ': '))  print(scores)if __name__ == '__main__':  main()
  names = ['关羽', '张飞', '赵云', '马超', '黄忠']
  subjs = ['语文', '数学', '英语']
  scores = [[0] * 3] * 5
  for row, name in enumerate(names):
    print('请输入%s的成绩' % name)
    for col, subj in enumerate(subjs):
      scores[row][col] = float(input(subj + ': '))
  print(scores)
if __name__ == '__main__':
  main()

我们希望录入5个学生3门课程的成绩,于是定义了一个有5个元素的列表,而列表中的每个元素又是一个由3个元素构成的列表,这样一个列表的列表刚好跟一个表格是一致的,相当于有5行3列。

接下来我们通过嵌套的for-in循环输入每个学生3门课程的成绩。程序执行完成后我们发现,每个学生3门课程的成绩是一模一样的(尴尬),而且就是最后录入的那个学生的成绩。

3。区分两个概念

要想把这个坑填平,我们首先要区分对象和对象的引用这两个概念,而要区分这两个概念,还得先说说内存中的栈和堆。

我们经常会听人说起“堆栈”这个词,但实际上“堆”和“栈”是两个不同的概念。众所周知,一个程序运行时需要占用一些内存空间来存储数据和代码,那么这些内存从逻辑上又可以做进一步的划分。

对底层语言(如C语言)有所了解的程序员大都知道,程序中可以使用的内存从逻辑上可以为五个部分,按照地址从高到低依次是:栈(stack)、堆(heap)、数据段(data segment)、只读数据段(static area)和代码段(code segment)。

栈用来存储局部、临时变量,以及函数调用时保存现场和恢复现场需要用到的数据,这部分内存在代码块开始执行时自动分配,代码块执行结束时自动释放,通常由编译器自动管理。

堆的大小不固定,可以动态的分配和回收,因此如果程序中有大量的数据需要处理,这些数据通常都放在堆上,如果堆空间没有正确的被释放会引发内存泄露的问题,而像Python、Java等编程语言都使用了垃圾回收机制来实现自动化的内存管理(自动回收不再使用的堆空间)。

4。小例子

所以,下面的代码中,变量a并不是真正的对象,它是对象的引用,相当于记录了对象在堆空间的地址,通过这个地址我们可以访问到对应的对象。

a = object()b = ['apple', 'pitaya', 'grape']
b = ['apple', 'pitaya', 'grape']

同理,变量b是列表容器的引用,它引用了堆空间上的列表容器,而列表容器中并没有保存真正的对象,它保存的也仅仅是对象的引用。

知道了这一点,我们可以回过头看看刚才的程序,我们对列表进行[[0]* 3] * 5操作时,仅仅是将[0, 0, 0] 这个列表的地址进行了复制,并没有创建新的列表对象。

所以,容器中虽然有5个元素,但是这5个元素引用了同一个列表对象。这一点可以通过id函数检查scores[0]和scores[1]的地址得到证实。在此我们举一个小例子,读者朋友们可以敲一敲加深印象。

a = [[0]*3]*5id(a[0])id(a[1])# id相等
id(a[1])
# id相等

5。正确代码

所以,正确的代码应该按照如下的方式进行修改。

def main():  names = ['关羽', '张飞', '赵云', '马超', '黄忠']  subjs = ['语文', '数学', '英语']  scores = [[]] * 5  for row, name in enumerate(names):    print('请输入%s的成绩' % name)    scores[row] = [0] * 3 #变为不再嵌套    for col, subj in enumerate(subjs):      scores[row][col] = float(input(subj + ': '))  print(scores)if __name__ == '__main__':  main()'关羽', '张飞', '赵云', '马超', '黄忠']
  subjs = ['语文', '数学', '英语']
  scores = [[]] * 5
  for row, name in enumerate(names):
    print('请输入%s的成绩' % name)
    scores[row] = [0] * 3 #变为不再嵌套
    for col, subj in enumerate(subjs):
      scores[row][col] = float(input(subj + ': '))
  print(scores)
if __name__ == '__main__':
  main()

或者

def main():  names = ['关羽', '张飞', '赵云', '马超', '黄忠']  subjs = ['语文', '数学', '英语']  scores = [[0] * 3 for _ in range(5)]  for row, name in enumerate(names):    print('请输入%s的成绩' % name)    scores[row] = [0] * 3    for col, subj in enumerate(subjs):      scores[row][col] = float(input(subj + ': '))  print(scores)if __name__ == '__main__':  main()
  names = ['关羽', '张飞', '赵云', '马超', '黄忠']
  subjs = ['语文', '数学', '英语']
  scores = [[0] * 3 for _ in range(5)]
  for row, name in enumerate(names):
    print('请输入%s的成绩' % name)
    scores[row] = [0] * 3
    for col, subj in enumerate(subjs):
      scores[row][col] = float(input(subj + ': '))
  print(scores)
if __name__ == '__main__':
  main()

以上就是使用嵌套列表需要注意的问题及解决措施,希望大家多多总结,以此避免在使用嵌套列表或者复制对象时可能遇到的坑。

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持三水点靠木。

Python 相关文章推荐
浅谈numpy库的常用基本操作方法
Jan 09 Python
Flask解决跨域的问题示例代码
Feb 12 Python
Python3调用百度AI识别图片中的文字功能示例【测试可用】
Mar 13 Python
Python基础教程之if判断,while循环,循环嵌套
Apr 25 Python
Python控制Firefox方法总结
Jun 03 Python
Python解析json时提示“string indices must be integers”问题解决方法
Jul 31 Python
Python 写入训练日志文件并控制台输出解析
Aug 13 Python
python求一个字符串的所有排列的实现方法
Feb 04 Python
Python Flask框架实现简单加法工具过程解析
Jun 03 Python
pytorch 限制GPU使用效率详解(计算效率)
Jun 27 Python
Python3爬虫中pyspider的安装步骤
Jul 29 Python
Requests什么的通通爬不了的Python超强反爬虫方案!
May 20 Python
python实现感知机模型的示例
Sep 30 #Python
python 实现关联规则算法Apriori的示例
Sep 30 #Python
Python之字典添加元素的几种方法
Sep 30 #Python
Python之字典对象的几种创建方法
Sep 30 #Python
python 实现朴素贝叶斯算法的示例
Sep 30 #Python
Python字典取键、值对的方法步骤
Sep 30 #Python
Python根据字典的值查询出对应的键的方法
Sep 30 #Python
You might like
PHP防注入安全代码
2008/04/09 PHP
基于HBase Thrift接口的一些使用问题及相关注意事项的详解
2013/06/03 PHP
使用php计算排列组合的方法
2013/11/13 PHP
php实现建立多层级目录的方法
2014/07/19 PHP
ThinkPHP中redirect用法分析
2014/12/05 PHP
关于PHP文件的自动运行方法分析
2016/05/13 PHP
ajax+php实现无刷新验证手机号的实例
2017/12/22 PHP
在JavaScript中使用inline函数的问题
2007/03/08 Javascript
浅析Js中的单引号与双引号问题
2013/11/06 Javascript
js表头排序实现方法
2015/01/16 Javascript
jquery实现左右滑动菜单效果代码
2015/08/27 Javascript
详解JavaScript 中的 replace 方法
2016/01/01 Javascript
JS使用单链表统计英语单词出现次数
2016/06/16 Javascript
JS数组操作之增删改查的简单实现
2017/08/21 Javascript
使用DataTable插件实现异步加载数据
2017/11/19 Javascript
jQuery实现表单动态加减、ajax表单提交功能
2018/06/08 jQuery
webpack的pitching loader详解
2019/09/23 Javascript
js实现搜索提示框效果
2020/09/05 Javascript
快速解决element的autofocus失效问题
2020/09/08 Javascript
vue3.0 项目搭建和使用流程
2021/03/04 Vue.js
python实现的udp协议Server和Client代码实例
2014/06/04 Python
python 根据正则表达式提取指定的内容实例详解
2016/12/04 Python
Python读取视频的两种方法(imageio和cv2)
2018/04/15 Python
PyQt实现界面翻转切换效果
2018/04/20 Python
python  创建一个保留重复值的列表的补码
2018/10/15 Python
从DataFrame中提取出Series或DataFrame对象的方法
2018/11/10 Python
Python3按一定数据位数格式处理bin文件的方法
2019/01/24 Python
OpenCV-Python 摄像头实时检测人脸代码实例
2019/04/30 Python
wxPython绘图模块wxPyPlot实现数据可视化
2019/11/19 Python
Python面向对象之私有属性和私有方法应用案例分析
2019/12/31 Python
浅谈selenium如何应对网页内容需要鼠标滚动加载的问题
2020/03/14 Python
keras的backend 设置 tensorflow,theano操作
2020/06/30 Python
利用python为PostgreSQL的表自动添加分区
2021/01/18 Python
标准大学生职业生涯规划书写作指南
2014/09/18 职场文书
2014年公路养护工作总结
2014/12/04 职场文书
2015年幼儿园师德师风建设工作总结
2015/10/23 职场文书