Python编程中犯的三种错误,让你浪费一下午时间

时间:2025-08-03  作者:Diven  阅读:0

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

Python编程中犯的三种错误,让你浪费一下午时间

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

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

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

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

def search_for_links(page, add_to=[]):

new_links = page.search_for_links()

add_to.extend(new_links)

return add_to

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

试试下面的代码:

def fn(var1, var2=[]):

var2.append(var1)

print var2

fn(3)

fn(4)

fn(5)

可能你认为我们将看到:

[3]

[4]

[5]

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

[3]

[3, 4]

[3, 4, 5]

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

fn(3, [4])

[4, 3]

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

def fn(var1, var2=None):

if not var2:

var2 = []

var2.append(var1)

或是在第一个例子中:

def search_for_links(page, add_to=None):

if not add_to:

add_to = []

new_links = page.search_for_links()

add_to.extend(new_links)

return add_to

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

def func(message=“my message”):

print message

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

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

class URLCatcher(object):

urls = []

def add_url(self, url):

self.urls.append(url)

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

a = URLCatcher()

a.add_url(‘http://www.google.com’)

b = URLCatcher()

b.add_url(‘http://www.bbc.co.hk’)

b.urls:

[‘http://www.google.com’, ‘http://www.bbc.co.uk’]

a.urls:

[‘http://www.google.com’, ‘http://www.bbc.co.uk’]

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

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

class URLCatcher(object):

def __init__(self):

self.urls = []

def add_url(self, url):

self.urls.append(url)

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

3、 可变的分配错误

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

a = {‘1’: “one”, ‘2’: ‘two’}

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

b = a

b[‘3’] = ‘three’

简单吧?

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

‘1’: “one”, ‘2’: ‘two’, ‘3’: ‘three’}

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

{‘1’: “one”, ‘2’: ‘two’, ‘3’: ‘three’}

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

c = (2, 3)

d = c

d = (4, 5)

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

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

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

b = a[:]

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

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

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

b = a.copy()

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

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

作者简介:

Pete Savage – Peter 是一位充满激情的开源爱好者,在过去十年里一直在推广和使用开源产品。从 Ubuntu 社区开始,在许多不同的领域自愿参与音频制作领域的研究工作。在职业经历方面,起初作为公司的系统管理员,大部分时间在管理和建立数据中心,之后在 Red Hat 担任 CloudForms 产品的主要测试工程师。

猜您喜欢

研磨膏是应用于金属加工、光学、电子等行业的材料,主要用于表面处理和抛光。根据不同的用途和成分,研磨膏可以分为几大类。按颗粒大小分类,研磨膏可分为粗研磨膏和细研磨...
2009-12-13 00:00:00

监测仪是应用于各个领域的高科技设备,能够实时收集和分析数据,为决策提供支持。在环境监测方面,监测仪可以用于空气质量、水质和土壤污染等数据的采集,帮助环保部门及时...
2011-04-19 00:00:00

中国企业主导,全球最大的电子产业链看上了这条MEMS赛道,产品有望加速普及!近日,全球首个MEMS扬声器行业标准——《MEMS 微型扬声器通用技术要求》团体标...
2024-08-08 16:37:00

蜂鸣片是应用于电子设备中的声响元件,主要用于发出警报声或提示音。根据不同的工作原理和结构,蜂鸣片可分为主动蜂鸣片和被动蜂鸣片两大类。主动蜂鸣片内置振荡电路,只需...
2011-10-05 00:00:00

射频模块是用于处理高频信号的电子组件,应用于通信、雷达、卫星、以及其无线应用中。基本功能是接收、放大、调制和解调射频信号,以确保信息的有效传输和接收。射频模块通...
2010-12-10 00:00:00

整形锉和圆锉是雕刻与修整工艺中不可少的工具,应用于木工、金属加工和模型制作等领域。整形锉通常用于对材料表面进行初步修整,其刀口设计能够快速去除多余的物料,使工件...
2009-08-03 00:00:00

现代电子设备和工业应用中,连接器是非常重要的配件。CONN-10P_18X12.8MM_TM是一款受到关注的连接器,因其很好的性能和可靠性而应用于各种场合。本文...
2025-04-24 10:31:12

稳压二极管是特殊的二极管,应用于电压稳压电路中。能够在一定的电压范围内保持稳定的输出电压,因此在电子产品中是重要的配件。正确识别稳压二极管的正负极是确保电路正常...
2025-04-02 22:01:07

贴片电阻是电子电路中不可或缺的元件,其阻值计算对于电路设计至关重要。这张实用公式表将帮助您快速掌握贴片电阻计算方法,轻松应对各种电路设计挑战。首先,我们需要了解...
2025-04-14 15:03:23

铝壳电阻因其优异的散热性能和稳定的电气特性,成为工业和电子设计中的重要组成部分。宇阳(EYANG)作为国内知名的电子元件品牌,其铝壳电阻产品因品质可靠、性能优越...
2022-11-07 19:09:03