Python开发者最常犯的10个错误

 Python是一门简单易学的编程语言,语法简洁而清晰,而且拥有丰富和强大的类库。与其它大多数程序设计语言使用大括号不同 ,它使用缩进来定义语句块。php

  在平时的工做中,Python开发者很容易犯一些小错误,这些错误都很容易避免,本文总结了Python开发者最常犯的10个错误,一块儿来看下,不知你中枪了没有。html

 

  1.滥用表达式做为函数参数默认值python

  Python容许开发者指定一个默认值给函数参数,虽然这是该语言的一个特征,但当参数可变时,很容易致使混乱,例如,下面这段函数定义:编程

>>> def foo(bar=[]):        # bar is optional and defaults to [] if not specified
...    bar.append("baz")    # but this line could be problematic, as we'll see... ... return bar

  在上面这段代码里,一旦重复调用foo()函数(没有指定一个bar参数),那么将一直返回'bar',由于没有指定参数,那么foo()每次被调用的时候,都会赋予[]。下面来看看,这样作的结果:闭包

>>> foo()
["baz"]
>>> foo()
["baz", "baz"]
>>> foo()
["baz", "baz", "baz"]

  解决方案:app

>>> def foo(bar=None):
...    if bar is None:		# or if not bar:
...        bar = []
...    bar.append("baz")
...    return bar
...
>>> foo()
["baz"]
>>> foo()
["baz"]
>>> foo()
["baz"]

  2.错误地使用类变量编程语言

  先看下面这个例子:ide

>>> class A(object):
...     x = 1
...
>>> class B(A):
...     pass
...
>>> class C(A):
...     pass
...
>>> print A.x, B.x, C.x
1 1 1

  这样是有意义的:函数

>>> B.x = 2
>>> print A.x, B.x, C.x
1 2 1

  再来一遍:工具

>>> A.x = 3
>>> print A.x, B.x, C.x
3 2 3

  仅仅是改变了A.x,为何C.x也跟着改变了。

  在Python中,类变量都是做为字典进行内部处理的,而且遵循方法解析顺序(MRO)。在上面这段代码中,由于属性x没有在类C中发现,它会查找它的基类(在上面例子中只有A,尽管Python支持多继承)。换句话说,就是C本身没有x属性,独立于A,所以,引用 C.x其实就是引用A.x。

  3.为异常指定不正确的参数

  假设代码中有以下代码:

>>> try:
...     l = ["a", "b"]
...     int(l[2])
... except ValueError, IndexError:  # To catch both exceptions, right?
...     pass
...
Traceback (most recent call last):
  File "<stdin>", line 3, in <module> IndexError: list index out of range

  问题在这里,except语句并不须要这种方式来指定异常列表。然而,在Python 2.x中,except Exception,e一般是用来绑定异常里的 第二参数,好让其进行更进一步的检查。所以,在上面这段代码里,IndexError异常并无被except语句捕获,异常最后被绑定 到了一个名叫IndexError的参数上。

  在一个异常语句里捕获多个异常的正确方法是指定第一个参数做为一个元组,该元组包含全部被捕获的异常。与此同时,使用as关键字来保证最大的可移植性,Python 2和Python 3都支持该语法。

>>> try:
...     l = ["a", "b"]
...     int(l[2])
... except (ValueError, IndexError) as e:  
...     pass
...
>>>

  4.误解Python规则范围

  Python的做用域解析是基于LEGB规则,分别是Local、Enclosing、Global、Built-in。实际上,这种解析方法也有一些玄机,看下面这个例子:

>>> x = 10
>>> def foo():
...     x += 1
...     print x
...
>>> foo()
Traceback (most recent call last):
  File "<stdin>", line 1, in <module>
  File "<stdin>", line 2, in foo
UnboundLocalError: local variable 'x' referenced before assignment

  许多人会感动惊讶,当他们在工做的函数体里添加一个参数语句,会在先前工做的代码里报UnboundLocalError错误( 点击这里查看更详细描述)。

  在使用列表时,开发者是很容易犯这种错误的,看看下面这个例子:

>>> lst = [1, 2, 3]
>>> def foo1():
...     lst.append(5)   # This works ok...
...
>>> foo1()
>>> lst
[1, 2, 3, 5]

>>> lst = [1, 2, 3]
>>> def foo2():
...     lst += [5]      # ... but this bombs!
...
>>> foo2()
Traceback (most recent call last):
  File "<stdin>", line 1, in <module> File "<stdin>", line 2, in foo UnboundLocalError: local variable 'lst' referenced before assignment

  为何foo2失败而foo1运行正常?

  答案与前面那个例子是同样的,但又有一些微妙之处。foo1没有赋值给lst,而foo2赋值了。lst += [5]实际上就是lst = lst + [5],试图给lst赋值(所以,假设Python是在局部做用域里)。然而,咱们正在寻找指定给lst的值是基于lst自己,其实还没有肯定。

  5.修改遍历列表

  下面这段代码很明显是错误的:

>>> odd = lambda x : bool(x % 2)
>>> numbers = [n for n in range(10)]
>>> for i in range(len(numbers)):
...     if odd(numbers[i]):
...         del numbers[i]  # BAD: Deleting item from a list while iterating over it
...
Traceback (most recent call last):
  	  File "<stdin>", line 2, in <module> IndexError: list index out of range

  在遍历的时候,对列表进行删除操做,这是很低级的错误。稍微有点经验的人都不会犯。

  对上面的代码进行修改,正确地执行:

>>> odd = lambda x : bool(x % 2)
>>> numbers = [n for n in range(10)]
>>> numbers[:] = [n for n in numbers if not odd(n)]  # ahh, the beauty of it all
>>> numbers
[0, 2, 4, 6, 8]

  6.如何在闭包中绑定变量

  看下面这个例子:

>>> def create_multipliers():
...     return [lambda x : i * x for i in range(5)]
>>> for multiplier in create_multipliers():
...     print multiplier(2)
...

  你指望的结果是:

<code>0
2
4
6
8</code>

  实际上:

<code>8
8
8
8
8</code>

  是否是很是吃惊!出现这种状况主要是由于Python的后期绑定行为,该变量在闭包中使用的同时,内部函数又在调用它。

  解决方案:

>>> 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

  7.建立循环模块依赖关系

  假设有两个文件,a.py和b.py,而后各自导入,以下:

  在a.py中:

import b

def f():
    return b.x
	
print f()

  在b.py中:

import a

x = 1

def g():
    print a.f()

首先,让咱们试着导入a.py:

<code>>>> import a
1</code>

  能够很好地工做,也许你会感到惊讶。毕竟,咱们确实在这里作了一个循环导入,难道不该该有点问题吗?

  仅仅存在一个循环导入并非Python自己问题,若是一个模块被导入,Python就不会试图从新导入。根据这一点,每一个模块在试图访问函数或变量时,可能会在运行时遇到些问题。

  当咱们试图导入b.py会发生什么(先前没有导入a.py):

>>> 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'

  出错了,这里的问题是,在导入b.py的过程当中还要试图导入a.py,这样就要调用f(),而且试图访问b.x。可是b.x并未被定义。

  能够这样解决,仅仅修改b.py导入到a.py中的g()函数

x = 1
def g():
    import a	# This will be evaluated only when g() is called
    print a.f()

  不管什么时候导入,一切均可以正常运行:

>>> 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'

  8.与Python标准库模块名称冲突

  Python拥有很是丰富的模块库,而且支持“开箱即用”。所以,若是不刻意避免,很容易发生命名冲突事件。例如,在你的代码中可能有一个email.py的模块,因为名称一致,它颇有可能与Python自带的标准库模块发生冲突。

  9.未按规定处理Python2.x和Python3.x之间的区别

  看一下foo.py:

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()

  在Python 2里面能够很好地运行:

$ python foo.py 1
key error
1
$ python foo.py 2
value error
2

  可是在Python 3里:

$ 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

  解决方案:

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()

  在Py3k中运行结果:

<code>$ python3 foo.py 1
key error
1
$ python3 foo.py 2
value error
2</code>

  在 Python招聘指南里有许多关于Python 2与Python 3在移植代码时须要关注的注意事项与讨论,你们能够前往看看。

  10.滥用__del__方法

  好比这里有一个叫mod.py的文件:

import foo
class Bar(object):
   	    ...
    def __del__(self):
        foo.cleanup(self.myhandle)

  下面,你在another_mod.py文件里执行以下操做:

import mod
mybar = mod.Bar()

  你会得到一个AttributeError异常。

  至于为何会出现该异常,点击这里查看详情。当解释器关闭时,该模块的全局变量所有设置为None。所以,在上面这个例子里,当__del__被调用时,foo已经所有被设置为None。

  一个很好的解决办法是使用atexit.register()代替。顺便说一句,当程序执行完成后,您注册的处理程序会在解释器关闭以前中止 工做。

  修复上面问题的代码:

import foo
import atexit

def cleanup(handle):
    foo.cleanup(handle)


class Bar(object):
    def __init__(self):
        ...
        atexit.register(cleanup, self.myhandle)

  在程序的正常终止的前提下,这个实现提供了一个整洁可靠的方式调用任何须要清理的功能。

  总结

  Python是一款强大而灵活的编程语言,而且带有许多机制和模式来大大提升工做效率。正如任何一门语言或软件工具同样,人们对其能力都会存在一个限制性地理解或欣赏,有些是弊大于利,有些时候反而会带来一些陷进。 体会一名语言的细微之处,理解一些常见的陷阱,有助于你在开发者的道路上走的更远。

相关文章
相关标签/搜索