源码网商城,靠谱的源码在线交易网站 我的订单 购物车 帮助

源码网商城

Python程序员开发中常犯的10个错误

  • 时间:2022-04-13 10:29 编辑: 来源: 阅读:
  • 扫一扫,手机访问
摘要:Python程序员开发中常犯的10个错误
Python是一门简单易学的编程语言,语法简洁而清晰,并且拥有丰富和强大的类库。与其它大多数程序设计语言使用大括号不一样 ,它使用缩进来定义语句块。   在平时的工作中,Python开发者很容易犯一些小错误,这些错误都很容易避免,本文总结了Python开发者最常犯的10个错误,一起来看下,不知你中枪了没有。 [b]1.滥用表达式作为函数参数默认值[/b]   Python允许开发者指定一个默认值给函数参数,虽然这是该语言的一个特征,但当参数可变时,很容易导致混乱,例如,下面这段函数定义:
[url=https://docs.python.org/2/faq/programming.html#why-am-i-getting-an-unboundlocalerror-when-the-variable-has-a-value]点击这里[/url]查看更详细描述)。 在使用列表时,开发者是很容易犯这种错误的,看看下面这个例子:
[u]复制代码[/u] 代码如下:
>>> 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本身,其实尚未确定。 [b]5.修改遍历列表[/b]   下面这段代码很明显是错误的:
[u]复制代码[/u] 代码如下:
>>> 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
在遍历的时候,对列表进行删除操作,这是很低级的错误。稍微有点经验的人都不会犯。   对上面的代码进行修改,正确地执行:
[u]复制代码[/u] 代码如下:
>>> 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]
[b]6.如何在闭包中绑定变量[/b]   看下面这个例子:
[u]复制代码[/u] 代码如下:
>>> def create_multipliers(): ...     return [lambda x : i * x for i in range(5)] >>> for multiplier in create_multipliers(): ...     print multiplier(2) ...
  你期望的结果是:
[u]复制代码[/u] 代码如下:
0 2 4 6 8
  实际上:
[u]复制代码[/u] 代码如下:
8 8 8 8 8
 是不是非常吃惊!出现这种情况主要是因为Python的后期绑定行为,该变量在闭包中使用的同时,内部函数又在调用它。   解决方案:
[u]复制代码[/u] 代码如下:
>>> 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
[b]7.创建循环模块依赖关系[/b]   假设有两个文件,a.py和b.py,然后各自导入,如下:   在a.py中:
[u]复制代码[/u] 代码如下:
import b def f():     return b.x   print f()
  在b.py中:
[u]复制代码[/u] 代码如下:
import a x = 1 def g():     print a.f()
首先,让我们试着导入a.py:
[u]复制代码[/u] 代码如下:
>>> import a 1
  可以很好地工作,也许你会感到惊讶。毕竟,我们确实在这里做了一个循环导入,难道不应该有点问题吗?   仅仅存在一个循环导入并不是Python本身问题,如果一个模块被导入,Python就不会试图重新导入。根据这一点,每个模块在试图访问函数或变量时,可能会在运行时遇到些问题。   当我们试图导入b.py会发生什么(先前没有导入a.py):
[u]复制代码[/u] 代码如下:
>>> 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()函数:
[u]复制代码[/u] 代码如下:
x = 1 def g():     import a # This will be evaluated only when g() is called     print a.f()
无论何时导入,一切都可以正常运行:
[u]复制代码[/u] 代码如下:
>>> 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'
[b]8.与Python标准库模块名称冲突[/b]   Python拥有非常丰富的模块库,并且支持“开箱即用”。因此,如果不刻意避免,很容易发生命名冲突事件。例如,在你的代码中可能有一个email.py的模块,由于名称一致,它很有可能与Python自带的标准库模块发生冲突。 [b]9.未按规定处理Python2.x和Python3.x之间的区别[/b]   看一下foo.py:
[u]复制代码[/u] 代码如下:
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里面可以很好地运行:
[u]复制代码[/u] 代码如下:
$ python foo.py 1 key error 1 $ python foo.py 2 value error 2
  但是在Python 3里:
[u]复制代码[/u] 代码如下:
$ 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
解决方案:
[u]复制代码[/u] 代码如下:
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中运行结果:
[u]复制代码[/u] 代码如下:
$ python3 foo.py 1 key error 1 $ python3 foo.py 2 value error 2
  在 Python招聘指南里有许多关于Python 2与Python 3在移植代码时需要关注的注意事项与讨论,大家可以前往看看。 [b]10.滥用__del__方法[/b]   比如这里有一个叫mod.py的文件:
[u]复制代码[/u] 代码如下:
import foo class Bar(object):         ...     def __del__(self):         foo.cleanup(self.myhandle)
  下面,你在another_mod.py文件里执行如下操作:
[u]复制代码[/u] 代码如下:
import mod mybar = mod.Bar()
 你会获得一个AttributeError异常。   至于为什么会出现该异常,点击这里查看详情。当解释器关闭时,该模块的全局变量全部设置为None。因此,在上面这个例子里,当__del__被调用时,foo已经全部被设置为None。   一个很好的解决办法是使用atexit.register()代替。顺便说一句,当程序执行完成后,您注册的处理程序会在解释器关闭之前停止 工作。   修复上面问题的代码:
[u]复制代码[/u] 代码如下:
import foo import atexit def cleanup(handle):     foo.cleanup(handle) class Bar(object):     def __init__(self):         ...         atexit.register(cleanup, self.myhandle)
在程序的正常终止的前提下,这个实现提供了一个整洁可靠的方式调用任何需要清理的功能。 [b]总结[/b]   Python是一款强大而灵活的编程语言,并且带有许多机制和模式来大大提高工作效率。正如任何一门语言或软件工具一样,人们对其能力都会存在一个限制性地理解或欣赏,有些是弊大于利,有些时候反而会带来一些陷进。 体会一名语言的细微之处,理解一些常见的陷阱,有助于你在开发者的道路上走的更远。
  • 全部评论(0)
联系客服
客服电话:
400-000-3129
微信版

扫一扫进微信版
返回顶部