说三道四技术文摘-感悟人生的经典句子
说三道四 > 文档快照

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

HTML文档下载 WORD文档下载 PDF文档下载
Python是一门简单易学的编程语言,语法简洁而清晰,并且拥有丰富和强大的类库。在日常开发中,开发者很容犯一些低级的错误,本文总结了开发者最容易犯的10个错误。

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

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


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

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"]
解决方案:

>>> 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.错误地使用类变量

先看下面这个例子:

>>> class A(object):...     x = 1...>>> class B(A):...     pass...>>> class C(A):...     pass...>>> print A.x, B.x, C.x1 1 1
这样是有意义的:

>>> B.x = 2>>> print A.x, B.x, C.x1 2 1
再来一遍:

>>> A.x = 3>>> print A.x, B.x, C.x3 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 fooUnboundLocalError: 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 fooUnboundLocalError: 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>02468</code>
实际上:

<code>88888</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)...02468
7.创建循环模块依赖关系

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

在a.py中:

import bdef f():    return b.x	print f()
在b.py中:

import ax = 1def g():    print a.f()
首先,让我们试着导入a.py:

<code>>>> import a1</code>

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

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

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

>>> import bTraceback (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.xAttributeError: 'module' object has no attribute 'x'

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

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

x = 1def 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 end1	# 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 sysdef 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 1key error1$ python foo.py 2value error2
但是在Python 3里:

$ python3 foo.py 1key errorTraceback (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 sysdef 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 1key error1$ python3 foo.py 2value error2</code>
在 Python招聘指南里有许多关于Python 2与Python 3在移植代码时需要关注的注意事项与讨论,大家可以前往看看。

10.滥用__del__方法

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

import fooclass Bar(object):   	    ...    def __del__(self):        foo.cleanup(self.myhandle)
下面,你在another_mod.py文件里执行如下操作:

import modmybar = mod.Bar()
你会获得一个AttributeError异常。

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

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

修复上面问题的代码:

import fooimport atexitdef cleanup(handle):    foo.cleanup(handle)class Bar(object):    def __init__(self):        ...        atexit.register(cleanup, self.myhandle)
在程序的正常终止的前提下,这个实现提供了一个整洁可靠的方式调用任何需要清理的功能。

总结

UNITE 2015 BEIJING 20大议题首爆 死路一条?智能硬件免费没有未来 APICloud推出“模块 Store” 公布APP创业 “Time to Market”战略 极路由加入海尔U+联盟 可跨品牌控制多款家居设备 《近匠》Amaze UI:做最懂中国程序员的开源HTML5框架 Testin CTO徐琨:要做测试领域的阿里云! TIOBE 2015年2月编程语言排行榜:JavaScript排名达历史最高点 机智云发布第二代GoKit开发板,兼容Arduino与Intel Edison 计算机视觉,让冰冷的机器看懂这个多彩的世界 昔日学霸反哺教育 打造教育领域今日头条 第三方DevOps实践报告发布 84%中国企业预将受益 《程序员必读之软件架构》作者Simon Brown:架构师与程序员的区别 Xcode 6.3 Beta发布,Swift 1.2带来哪些新变化? 应用生与死:移动测试究竟该怎么搞? Chrome将跟SPDY说再见,拥抱HTTP/2 ImageNet挑战赛中超越人类的计算机视觉系统 微软发布首款Android可穿戴设备人工智能产品 红包大战!双11过后,微信支付宝的又一个营销游戏 O2O大潮来袭,洗牌将至? 谈谈UI架构设计的演化 Node.js 开源基金会成立,Joyent让出领导权 LinkedIn分布式实时性能跟踪与效率优化实例 DIY机器人不是事!软硬两不误的开发平台RoboCORE 详解OS X和iOS图像处理框架Core Image 快的打车联合创始人兼技术副总裁闻诚:CTO要有“334”能力 阿里巴巴正式加入W3C,成为W3C会员 在Swift中,如何像Objective-C定义可选接口? DARPA:4年内推出具备触觉的机械手 Silver:用Swift编写iOS、Android和.NET应用 Swift没有异常处理,遇到功能性错误怎么办? 基于Python的测试驱动开发实战 请问为什么sql server2000数据库里的数据在Win XP下不能显示中文啊?? 有社会保险个人帐户管理系统编写经验的请进来? 我想知道在OICQ上,我怎么能知道它有消息来了。。。可以得到它的消息吗。。怎样等到?? C++中的运算符重载在现实中有什么用径? csdn--哈尔滨的朋友来签下到,看看有多少 Sqlserver用ADO+Tdbgrid录入数据,不用触发器怎样实现对另一个表的增删改。 谁有C和C++标准库函数,求一份,地址内详 救命呀!关于visualage安装的问题 菜鸟想问SetLength和Move是什么意思,什么用法,SetLength(S, BufferLength); 和Move(Buffer^, PChar(S)^, BufferLength);是什么意思,高 我要把一份文档保存一个副本在本地怎么做? 有人用过IBM的JAVA开发工具么 求救!关于JNI 有人知道.sgf(棋类通用文件)的格式吗? 在JAVA下如何实现用户输入数据,像C的SCANF一样,给个例子吧 如何将一片缓冲区传入VC写的DLL中? 怎样在网页中插入视频文件asf,wmv格式的,要怎么写这条语句 **CSS的字体居中怎么控制** 求大家帮帮我:vc掉dll的问题 急需功能齐全的ZIP控件(要求免费或已破解的)高分相酬!!!!! 一个很菜的问题,请高手相助! VC中用COM如何传递一个大数组? ADODB.Recordset 错误 '800a0cb3' 请问:哪里有matlab的linux版本下载 初学者:这段程序在编译时为什么会出错?? 怎么知道一个光驱是可写的? 请问如何在vc中sdk方法建立数据库接口 为什么98出现许多乱码拉? ===菜鸟系列===:如何用VC获得www.csdn.net的IP地址?在线等候!!! 请教一道算法题!44的44次方!用C或C++实现! 关于文件上传的问题 lihonggen请进,多问1句 98二的CD-KEY给一个,马上给分 TC++图像编程小问题: ) 请教一下怎么解决~~ How to install VB6.0 Enterprise in Windows NT workstation4.0+sp6? ###########################简单的问题,救我! 关于open方法的问题??? =======是鲁迅错了?还是我错了?============= 如何实现写字板的replace功能? 新手问初级的问题:客户端使用什么数据库好? 为什么 CBC6 没有 ClassExplorer ??????? vc高手看过来!如何实现通过程序拨号(专线猫)请给出例子!急.急.急! 我的机器为何不能装2000 想知道搞Java多少银子/月呀? 哪里有Win2000英文版的下载啊 请教各位:用jsp中怎样知道用户浏览器的类型? 哪里有bmp文件格式的标准? 为什么sql server2000数据库里的数据在Win XP下不能显示中文啊? 回答出来给100分 在pb中用SetPicture(lb_image)当lb_image为空时picture的图片不变 急!!请问关于传递byte数组的问题 请教动态库的问题! X Y为实数 满足2X^2+2Y^2+3XY=1 求X^2+Y^2最大值 有哪些珍惜时间的格言 关于我国河流的叙述,正确的是A.我国大多数河流自西向东流入太平洋 B.东部地区的河流冬季普遍有结冰现象 C.西北广大地区的河流均为内流河 D.外流区的河流,越往北去汛期越长 已知不等式(k-1)x²+2x+1≥0对一切x∈R恒成立,则实数k的取值范围是 若关于x的方程m/x*x-x-2=x/x+1-x-1/x-2的解大于零,求m的取值范围 已知m为有理数,求m的绝对值 请问一下德州安心的强化地板怎么样,我想铺点强化地板 关于x的方程x+m/x-2=-1的根大于零,则m的取值范围是直接告诉结果. 设经过椭圆x^2/4+y^2/3=1的左焦点f1且倾斜角为45度的直线交该椭圆于ab两点求三角形abf2的周长求ab中点m的坐标 想买几十元一平方的强化木地板,哪里有,不能太差条件有限,铺客厅用的强化木地板,不能太差,不能太贵 已知关于X的方程X+a/X-2=-1的解大于零,求a的取值范围 有关于珍惜时间的 名言警句 强化地板怎么选择?决定买强化地板了家里装修决定买强化地板,但是品牌很多,如何挑选呢 混凝土中加入的青石是什么岩石破碎的?比普通卵石有什么不同? 若有理数a的相反数是1,有理数m的绝对值是1,求a+m的值. 强化地板得厚度有几种?判断强化地板的质量是看厚度还是看什么?上海哪家强化地板很标准,【罗马阳光地板】可以吗? 鹅卵石是不是岩石 鹅卵石可以透光,有价值吗 已知关于x的方程2x+m/x-3=-2的解为负数,求m的取值范围 鹅卵石属于哪种岩石?是沉积岩、岩浆岩还是变质岩? 已知椭圆的方程是X^2/16+Y^2/9=1,过左焦点F1的直线叫椭圆于A、B两点,求三角形ABF2的周长请不会的朋友别乱答、谢谢. 已知关于x的方程三分之x+m减二分之2x减1=m的解的负数,求m的取值范围 鹅卵石属于什么种类的岩石?RT 珍惜时间的名言警句自己写 还是写人家的?‘ 一块长方形的钢板,长是10米,宽是4米,每平方米重8千克,这块钢板重多少千克? 关于珍惜时间的名言 若有理数m满足:m的绝对值=m+1,则m+几 有一块平行四边形的钢板,高是0.9米,底是它的2倍.如果每平方米钢板重15.4千克 ,这块钢板重多少千克? 关于珍惜时间的名言警句 我有一块鹅卵石,能够透光的,想问一下价值多少左右? 已知实数x、y满足5x²-3xy+1/2y²-2x+1/2y+1/4=0 求x、y的值 混凝土采用矿渣水泥,卵石和天然砂配制而成,他耐温80-200度可以吗,具有防腐能力吗?如果不耐温、耐酸,还有什么最廉价的配方吗? 椭圆x^2/9+y^2/16=1的两焦点分别为F1,F2,过F1任意作直线交椭圆于A,B两点,则三角形ABF2的周长为多少? 初二上地理题 急等答案 矿渣水泥 火山灰水泥拌制的大体积混凝土的养护时间不得少于( )A 7天B 14天C 21天D 28天要说下为什么选那个,我只知道3天和28天两个数值,但28天好像水泥都硬的邦邦的了. 如果m是任意有理数,那么m+m的绝对值是?有以下答案:1.必为正数.2.必为负数 3.必为0 4.必为非负数 关于珍惜时间的名人名言 浇筑柱子混凝土时,其要部应先浇50~100MM厚的与所浇混凝土内水泥砂浆成分相同的水泥砂浆求原理?根部 已知椭圆两焦点为F1,F2,a=3/2,过F1作直线交椭圆于A,B两点,则△ABF2的周长为如题.拜托啦╭(╯3╰)╮ 混泥土C15 C25 大致是这样的问题:一月我国受来自北方寒潮影响,全国气温普遍低,秦岭以北的西安为-20℃,秦岭以南的安康气温为-7℃,分析原因,三条 已知α,β属于(0,π)且tanα,tanβ是方程x的平方-5x+6=0的两实根(1)求α+β的值(2) 求cos(α-β)的 《培根随笔》中的格言警句尽多 珍惜时间的名人名言 把8拆成若干个自然数的和,使这些自然数的乘积最大? 是否存在实数a对区间[0,3]上任意实数x不等式log(2a²+1)(2x+2) f1,f2是椭圆x^2/2+y^2=1的两个焦点,过f2作倾斜角为45度的直线AB于椭圆交于A,B两点,三角形f1AB的面积多少 关于X的分式方程 有解,求K的取值范围 若关于x的不等式|x +3|- |x -2|>=log 小2a 有解,则实数a 的取值范围是多少 已知椭圆x^2/2+y^2/1=1的左右焦点分别为F1,F2,若过点P(0,-2)及F1的直线交椭圆于A,B两点,求三角形ABF...已知椭圆x^2/2+y^2/1=1的左右焦点分别为F1,F2,若过点P(0,-2)及F1的直线交椭圆于A,B两点,求三角形ABF2 有49个人每人一号,号是1到49.挑选若干人排一圆圈,任何相邻的两个号的乘积小于100,最多能挑出多少个人 建筑用的一平方立米混泥土有多重(吨?) 已知椭圆方程为x^/16+y^/=1的左右焦点分别为f1,f2,过左焦点f1的直线交椭圆于A,B两点,求三角形ABf2的周长 有49个小孩,每人胸前有一个号码,号码从1到49各不相同.请你挑选若干个小孩,排成一个圆圈,使任何相邻两个小孩的号码的乘积小于100.你最多挑选出多少孩子? 直径1米的混泥土球有多重 初二地理题,高手解答 有49人 每人拿着数 数从1到49不相同 选出若干个人 排成圆圈 使任何相邻两个人数乘积小于100 最多选出多少还有:(2)在50个连续三位数中,三位数的三个数字之和能被7整除的数,最多有多少个? 混泥土一方大概多少吨 珍惜时间的格言有哪些? 已知m是绝对值最小的有理数,快啊!11已知m是绝对值最小的有理数,且-2a^(m+2)b^(y+1)与3a^xb^3是同类项,求多项式2x^3-3xy+6y^2-3mx^3-mxy-9my^2的值
备案号:鲁ICP备13029499号-2 说三道四 www.s3d4.cn 说三道四技术文摘