学习用Python编程时要避免的3个错误

这些错误会造成很麻烦的问题,需要数小时才能解决。

当你做错事时,承认错误并不是一件容易的事,但是犯错是任何学习过程中的一部分,无论是学习走路,还是学习一种新的编程语言都是这样,比如学习 Python。

为了让初学 Python 的程序员避免犯同样的错误,以下列出了我学习 Python 时犯的三种错误。这些错误要么是我长期以来经常犯的,要么是造成了需要几个小时解决的麻烦。

年轻的程序员们可要注意了,这些错误是会浪费一下午的!

1、 可变数据类型作为函数定义中的默认参数

这似乎是对的?你写了一个小函数,比如,搜索当前页面上的链接,并可选将其附加到另一个提供的列表中。


  1. def search_for_links(page, add_to=[]): 
  2.  
  3.     new_links = page.search_for_links() 
  4.  
  5.     add_to.extend(new_links) 
  6.  
  7.     return add_to  

从表面看,这像是十分正常的 Python 代码,事实上它也是,而且是可以运行的。但是,这里有个问题。如果我们给 add_to 参数提供了一个列表,它将按照我们预期的那样工作。但是,如果我们让它使用默认值,就会出现一些神奇的事情。

试试下面的代码:


  1. def fn(var1, var2=[]): 
  2.  
  3.     var2.append(var1) 
  4.  
  5.     print var2 
  6.  
  7. fn(3) 
  8.  
  9. fn(4) 
  10.  
  11. fn(5)  

可能你认为我们将看到:


  1. [3] 
  2.  
  3. [4] 
  4.  
  5. [5]  

但实际上,我们看到的却是:


  1. [3] 
  2.  
  3. [3, 4] 
  4.  
  5. [3, 4, 5]  

为什么呢?如你所见,每次都使用的是同一个列表,输出为什么会是这样?在 Python
中,当我们编写这样的函数时,这个列表被实例化为函数定义的一部分。当函数运行时,它并不是每次都被实例化。这意味着,这个函数会一直使用完全一样的列表对象,除非我们提供一个新的对象:


  1. fn(3, [4]) 

  1. [4, 3]  

答案正如我们所想的那样。要想得到这种结果,正确的方法是:


  1. def fn(var1, var2=None): 
  2.  
  3.     if not var2: 
  4.  
  5.         var2 = [] 
  6.  
  7.     var2.append(var1)  

或是在第一个例子中:


  1. def search_for_links(page, add_to=None): 
  2.  
  3.     if not add_to: 
  4.  
  5.         add_to = [] 
  6.  
  7.     new_links = page.search_for_links() 
  8.  
  9.     add_to.extend(new_links) 
  10.  
  11.     return add_to  

这将在模块加载的时候移走实例化的内容,以便每次运行函数时都会发生列表实例化。请注意,对于不可变数据类型,比如元组、字符串、整型,是不需要考虑这种情况的。这意味着,像下面这样的代码是非常可行的:


  1. def func(message="my message"): 
  2.  
  3. print message  

2、 可变数据类型作为类变量

这和上面提到的最后一个错误很相像。思考以下代码:


  1. class URLCatcher(object): 
  2.  
  3.     urls = [] 
  4.  
  5.     def add_url(self, url): 
  6.  
  7.         self.urls.append(url)  

这段代码看起来非常正常。我们有一个储存 URL 的对象。当我们调用 add_url 方法时,它会添加一个给定的 URL 到存储中。看起来非常正确吧?让我们看看实际是怎样的:


  1. a = URLCatcher() 
  2.  
  3. a.add_url('http://www.google.com') 
  4.  
  5. b = URLCatcher() 
  6.  
  7. b.add_url('http://www.bbc.co.hk')  

b.urls:


  1. ['http://www.google.com', 'http://www.bbc.co.uk'] 

a.urls:


  1. ['http://www.google.com', 'http://www.bbc.co.uk'] 

等等,怎么回事?!我们想的不是这样啊。我们实例化了两个单独的对象 a 和 b。把一个 URL 给了 a,另一个给了 b。这两个对象怎么会都有这两个 URL 呢?

这和第一个错例是同样的问题。创建类定义时,URL 列表将被实例化。该类所有的实例使用相同的列表。在有些时候这种情况是有用的,但大多数时候你并不想这样做。你希望每个对象有一个单独的储存。为此,我们修改代码为:


  1. class URLCatcher(object): 
  2.  
  3.     def __init__(self): 
  4.  
  5.         self.urls = [] 
  6.  
  7.     def add_url(self, url): 
  8.  
  9.         self.urls.append(url)  

现在,当创建对象时,URL 列表被实例化。当我们实例化两个单独的对象时,它们将分别使用两个单独的列表。

3、 可变的分配错误

这个问题困扰了我一段时间。让我们做出一些改变,并使用另一种可变数据类型 – 字典。


  1. a = {'1': "one", '2': 'two'} 

现在,假设我们想把这个字典用在别的地方,且保持它的初始数据完整。


  1. b = a 
  2.  
  3. b['3'] = 'three'  

简单吧?

现在,让我们看看原来那个我们不想改变的字典 a:


  1. {'1': "one", '2': 'two', '3': 'three'} 

哇等一下,我们再看看 b?


  1. {'1': "one", '2': 'two', '3': 'three'} 

等等,什么?有点乱……让我们回想一下,看看其它不可变类型在这种情况下会发生什么,例如一个元组:


  1. c = (2, 3) 
  2.  
  3. d = c 
  4.  
  5. d = (4, 5)  

现在 c 是 (2, 3),而 d 是 (4, 5)。

这个函数结果如我们所料。那么,在之前的例子中到底发生了什么?当使用可变类型时,其行为有点像 C 语言的一个指针。在上面的代码中,我们令 b
= a,我们真正表达的意思是:b 成为 a 的一个引用。它们都指向 Python
内存中的同一个对象。听起来有些熟悉?那是因为这个问题与先前的相似。其实,这篇文章应该被称为「可变引发的麻烦」。

列表也会发生同样的事吗?是的。那么我们如何解决呢?这必须非常小心。如果我们真的需要复制一个列表进行处理,我们可以这样做:


  1. b = a[:] 

这将遍历并复制列表中的每个对象的引用,并且把它放在一个新的列表中。但是要注意:如果列表中的每个对象都是可变的,我们将再次获得它们的引用,而不是完整的副本。

假设在一张纸上列清单。在原来的例子中相当于,A 某和 B
某正在看着同一张纸。如果有个人修改了这个清单,两个人都将看到相同的变化。当我们复制引用时,每个人现在有了他们自己的清单。但是,我们假设这个清单包括寻找食物的地方。如果“冰箱”是列表中的第一个,即使它被复制,两个列表中的条目也都指向同一个冰箱。所以,如果冰箱被
A 修改,吃掉了里面的大蛋糕,B
也将看到这个蛋糕的消失。这里没有简单的方法解决它。只要你记住它,并编写代码的时候,使用不会造成这个问题的方式。

字典以相同的方式工作,并且你可以通过以下方式创建一个昂贵副本:


  1. b = a.copy() 

再次说明,这只会创建一个新的字典,指向原来存在的相同的条目。因此,如果我们有两个相同的列表,并且我们修改字典 a 的一个键指向的可变对象,那么在字典 b 中也将看到这些变化。

可变数据类型的麻烦也是它们强大的地方。以上都不是实际中的问题;它们是一些要注意防止出现的问题。在第三个项目中使用昂贵复制操作作为解决方案在 99% 的时候是没有必要的。你的程序或许应该被改改,所以在第一个例子中,这些副本甚至是不需要的。

作者:佚名

来源:51CTO

时间: 2024-10-02 18:49:53

学习用Python编程时要避免的3个错误的相关文章

分析Python编程时利用wxPython来支持多线程的方法_python

如果你经常使用python开发GUI程序的话,那么就知道,有时你需要很长时间来执行一个任务.当然,如果你使用命令行程序来做的话,你回非常惊讶.大部分情况下,这会堵塞GUI的事件循环,用户会看到程序卡死.如何才能避免这种情况呢?当然是利用线程或进程了!本文,我们将探索如何使用wxPython和theading模块来实现. wxpython线程安全方法 wxPython中,有三个"线程安全"的函数.如果你在更新UI界面时,三个函数都不使用,那么你可能会遇到奇怪的问题.有时GUI也忙运行挺正

《趣学Python编程》——术语表

术语表 当你刚开始编程的时候,你会遇到不太理解的术语.这种不理解会成为你进步的阻碍.但是这很好办! 我创建了下面的术语表来帮助你解释这些单词和术语.在里边有很多本书中用到的编程术语的定义,如果你遇到了不理解的东西就到这里来找一找吧. 动画(animation) 以足够快的速度把图片显示出来的过程,看上去就像在动. 语句块(block) 计算机程序中的一组语句. 布尔(Boolean) 非真即假的一种值.(在Python里是True或False,其中的T和F都要大写.) 调用(call) 运行函数

《树莓派Python编程入门与实战》——第一部分 树莓派编程环境 第1章 配置树莓派 1.1 树莓派是什么

第一部分 树莓派编程环境 第1章 配置树莓派 第2章 认识Raspbian linux发行版 第3章 搭建编程环境 第1章 配置树莓派 在本章中,你将学习如下内容. 树莓派是什么 如何获得一个树莓派 你的树莓派可能需要的一些外设 如何让树莓派工作 如何排除树莓派的故障 本章主要介绍树莓派:它是什么,它的历史,以及为什么你需要学习用Python在树莓派上进行编程.最后,你将了解到一些树莓派的外设以及将这些外设与树莓派组装好并运行起来的方法. 1.1 树莓派是什么 树莓派是一个非常廉价的.只有手掌大

《树莓派Python编程入门与实战(第2版)》——第一部分 树莓派编程环境 第1章 配置树莓派 1.1 获取树莓派

第一部分 树莓派编程环境 第1章 配置树莓派 第2章 认识Raspbian Linux发行版 第3章 搭建编程环境 第1章 配置树莓派 本章主要内容包括: 树莓派是什么 如何获得一个树莓派 你的树莓派可能需要的一些外围设备 如何让树莓派工作 如何排除树莓派的故障 本章主要介绍树莓派:它是什么,它的历史,以及为什么需要学习用Python在树莓派上编程.最后,你将了解到一些树莓派的外围设备以及将这些外围设备与树莓派组装好并运行起来的方法. 1.1 获取树莓派 树莓派是一个非常便宜的.只有手掌大小的完

《树莓派Python编程入门与实战》——1.1 树莓派是什么

1.1 树莓派是什么 树莓派Python编程入门与实战树莓派是一个非常廉价的.只有手掌大小的完全可编程的计算机(见图1.1).虽然树莓派的体积小,但是它的潜力无限.你可以像使用常规台式计算机一样在树莓派上创建一个非常酷的工程.例如,你可以用树莓派搭建你自己的家用云存储服务器. 1.1.1 树莓派的历史 树莓派仍然是一个相当年轻的装置.它是由Eben Upton和几个同事在英国发明的.它的第一个商业版本(A)型在2012年初以25美元的低价正式发售. 提示: 树莓派的不同简称 人们经常使用不同的名

《树莓派Python编程入门与实战(第2版)》——导读

前 言 2012年2月一经官方首发,树莓派就在全球引起了一阵旋风,10000套设备瞬间售罄.它是一个廉价的.只有信用卡大小的裸露电路板,同时,它是一个运行开源Linux操作系统的完全可编程的PC系统.树莓派可以连接到互联网上,可以插到电视上,并且其最新的第2版采用一个很快的ARM处理器,其性能可以与很多平板设备匹敌,而这一切仅需35美元. 树莓派最初只是为了激发学龄儿童对计算机的兴趣,但是它在世界范围内引起了极客.企业家和教育家的广泛关注.截至2015年6月,销售了600万台左右. 树莓派的官方

《树莓派Python编程入门与实战》——导读

前 言 树莓派于2012年2月一经官方首发就在全球引起了一阵旋风,10000套设备瞬间售罄.它是一个廉价的只有信用卡大小的裸露电路板,同时,它是一个运行开源Linux操作系统的完全可编程的PC系统.树莓派可以连接到互联网上,可以插到电视上,并且它仅需35美元. 树莓派最初只是为了激发学龄儿童对计算机的兴趣,但是它在世界范围内引起了极客.企业家和教育家的广泛关注. 树莓派的官方编程语言是Python.Python是一种灵活的编程语言,可以运行在任何平台上.因此,程序可以在Windows PC或者M

《树莓派Python编程入门与实战》——第1章 配置树莓派

第1章 配置树莓派 树莓派Python编程入门与实战在本章中,你将学习如下内容. 树莓派是什么如何获得一个树莓派你的树莓派可能需要的一些外设如何让树莓派工作如何排除树莓派的故障本章主要介绍树莓派:它是什么,它的历史,以及为什么你需要学习用Python在树莓派上进行编程.最后,你将了解到一些树莓派的外设以及将这些外设与树莓派组装好并运行起来的方法.

《树莓派Python编程入门与实战(第2版)》——3.6 关于Python开发环境shell

3.6 关于Python开发环境shell 开发环境shell是用户创建.运行.测试和修改Python脚本的工具.通常开发环境会改变代码关键语法的颜色,以便更容易识别各种语句.这种颜色标注,有利于脚本的测试.修改以及调试.另一个不错的功能是代码自动完成,当输入Python关键字时,开发环境会提供一些屏幕提示来帮助你完成代码. 除此之外,开发环境还提供语法检查,因此你可以在不运行整个Python脚本的情况下就检查出语法错误.通常,开发环境工具还提供了自动缩进来保持整个脚本的缩进一致. 最后,环境中