个人最新博客地址http://www.skylway.com/
分类: Python/Ruby
2014-05-22 17:27:19
原文地址:python开发者容易犯的10个错误 作者:刘一痕
Python是一门简单易学的编程语言,语法简洁而清晰,并且拥有丰富和强大的类库。与其它大多数程序设计语言使用大括号不一样 ,它使用缩进来定义语句块。
在平时的工作中,Python开发者很容易犯一些小错误,这些错误都很容易避免,本文总结了Python开发者最常犯的10个错误,一起来看下,不知你中枪了没有。
1.滥用表达式作为函数参数默认值
Python允许开发者指定一个默认值给函数参数,虽然这是该语言的一个特征,但当参数可变时,很容易导致混乱,例如,下面这段函数定义:
在上面这段代码里,一旦重复调用foo()函数(没有指定一个bar参数),那么将一直返回'bar',因为没有指定参数,那么foo()每次被调用的时候,都会赋予[]。下面来看看,这样做的结果:
解决方案:
先看下面这个例子:
在Python中,类变量都是作为字典进行内部处理的,并且遵循方法解析顺序(MRO)。在上面这段代码中,因为属性x没有在类C中发现,它会查找它的基类(在上面例子中只有A,尽管Python支持多继承)。换句话说,就是C自己没有x属性,独立于A,因此,引用 C.x其实就是引用A.x。
3.为异常指定不正确的参数
假设代码中有如下代码:
问题在这里,except语句并不需要这种方式来指定异常列表。然而,在Python 2.x中,except Exception,e通常是用来绑定异常里的 第二参数,好让其进行更进一步的检查。因此,在上面这段代码里,IndexError异常并没有被except语句捕获,异常最后被绑定 到了一个名叫IndexError的参数上。
在一个异常语句里捕获多个异常的正确方法是指定第一个参数作为一个元组,该元组包含所有被捕获的异常。与此同时,使用as关键字来保证最大的可移植性,Python 2和Python 3都支持该语法。
4.误解Python规则范围
Python的作用域解析是基于LEGB规则,分别是Local、Enclosing、Global、Built-in。实际上,这种解析方法也有一些玄机,看下面这个例子:
许多人会感动惊讶,当他们在工作的函数体里添加一个参数语句,会在先前工作的代码里报UnboundLocalError错误( 点击这里查看更详细描述)。
在使用列表时,开发者是很容易犯这种错误的,看看下面这个例子:
答案与前面那个例子是一样的,但又有一些微妙之处。foo1没有赋值给lst,而foo2赋值了。lst += [5]实际上就是lst = lst + [5],试图给lst赋值(因此,假设Python是在局部作用域里)。然而,我们正在寻找指定给lst的值是基于lst本身,其实尚未确定。
5.修改遍历列表
下面这段代码很明显是错误的:
对上面的代码进行修改,正确地执行:
看下面这个例子:
解决方案:
1 2 3 4 5 6 7 8 9 10 11 | >>> def create_multipliers(): ... return [ lambda x, i = i : i * x for i in range ( 5 )] ... >>> for multiplier in create_multipliers(): ... print multiplier( 2 ) ... 0 2 4 6 8 |
假设有两个文件,a.py和b.py,然后各自导入,如下:
在a.py中:
1 2 3 4 5 6 | import b def f(): return b.x print f() |
1 2 3 4 5 6 | import a x = 1 def g(): print a.f() |
可以很好地工作,也许你会感到惊讶。毕竟,我们确实在这里做了一个循环导入,难道不应该有点问题吗?
仅仅存在一个循环导入并不是Python本身问题,如果一个模块被导入,Python就不会试图重新导入。根据这一点,每个模块在试图访问函数或变量时,可能会在运行时遇到些问题。
当我们试图导入b.py会发生什么(先前没有导入a.py):
1 2 3 4 5 6 7 8 9 10 | >>> import b Traceback (most recent call last): File "<stdin>" , line 1 , in <module> File "b.py" , line 1 , in <module> import a File "a.py" , line 6 , in <module> print f() File "a.py" , line 4 , in f return b.x AttributeError: 'module' object has no attribute 'x' < / module>< / module>< / module>< / stdin> |
出错了,这里的问题是,在导入b.py的过程中还要试图导入a.py,这样就要调用f(),并且试图访问b.x。但是b.x并未被定义。
可以这样解决,仅仅修改b.py导入到a.py中的g()函数:
1 2 3 4 | x = 1 def g(): import a # This will be evaluated only when g() is called print a.f() |
1 2 3 4 | >>> import b >>> b.g() 1 # Printed a first time since module 'a' calls 'print f()' at the end 1 # Printed a second time, this one is our call to 'g' |
Python拥有非常丰富的模块库,并且支持“开箱即用”。因此,如果不刻意避免,很容易发生命名冲突事件。例如,在你的代码中可能有一个email.py的模块,由于名称一致,它很有可能与Python自带的标准库模块发生冲突。
9.未按规定处理Python2.x和Python3.x之间的区别
看一下foo.py:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 | import sys def bar(i): if i = = 1 : raise KeyError( 1 ) if i = = 2 : raise ValueError( 2 ) def bad(): e = None try : bar( int (sys.argv[ 1 ])) except KeyError as e: print ( 'key error' ) except ValueError as e: print ( 'value error' ) print (e) bad() |
1 2 3 4 5 6 | $ python foo.py 1 key error 1 $ python foo.py 2 value error 2 |
1 2 3 4 5 6 7 8 | $ python3 foo.py 1 key error Traceback (most recent call last): File "foo.py" , line 19 , in <module> bad() File "foo.py" , line 17 , in bad print (e) UnboundLocalError: local variable 'e' referenced before assignment< / module> |
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 | import sys def bar(i): if i = = 1 : raise KeyError( 1 ) if i = = 2 : raise ValueError( 2 ) def good(): exception = None try : bar( int (sys.argv[ 1 ])) except KeyError as e: exception = e print ( 'key error' ) except ValueError as e: exception = e print ( 'value error' ) print (exception) good() |
10.滥用__del__方法
比如这里有一个叫mod.py的文件:
1 2 3 4 5 | import foo class Bar( object ): ... def __del__( self ): foo.cleanup( self .myhandle) |
1 2 | import mod mybar = mod.Bar() |
至于为什么会出现该异常,点击这里查看详情。当解释器关闭时,该模块的全局变量全部设置为None。因此,在上面这个例子里,当__del__被调用时,foo已经全部被设置为None。
一个很好的解决办法是使用atexit.register()代替。顺便说一句,当程序执行完成后,您注册的处理程序会在解释器关闭之前停止 工作。
修复上面问题的代码:
1 2 3 4 5 6 7 8 9 10 11 | import foo import atexit def cleanup(handle): foo.cleanup(handle) class Bar( object ): def __init__( self ): ... atexit.register(cleanup, self .myhandle) |
Python是一款强大而灵活的编程语言,并且带有许多机制和模式来大大提高工作效率。正如任何一门语言或软件工具一样,人们对其能力都会存在一个限制性地理解或欣赏,有些是弊大于利,有些时候反而会带来一些陷进。 体会一名语言的细微之处,理解一些常见地陷进,有助于你在开发者的道路上走的更远。