python – 为什么设置一个默认参数值使这个函数成为一个闭包?

我正在编写一个应用程序,其中标签是可链接的,并且需要检索整个链接标签链.不允许自我引用.运行以下代码最终会得到一些非常奇怪的结果:

class Tag(object):
  def __init__(self,name):
    self.name = name
    self.links = []

  def __repr__(self):
    return "<Tag {0}>".format(self.name)

  def link(self,tag):
    self.links.append(tag)


def tag_chain(tag,kNown=[]):
  chain = []
  if tag not in kNown:
    kNown.append(tag)
  print "KNown: {0}".format(kNown)

  for link in tag.links:
    if link in kNown:
      continue
    else:
      kNown.append(link)
    chain.append(link)
    chain.extend(tag_chain(link,kNown))
  return chain

a = Tag("a")
b = Tag("b")
c = Tag("c")
a.link(b)
b.link(c)
c.link(a)

o = tag_chain(a)
print "Result:",o
print "------------------"
o = tag_chain(a)
print "Result:",o

结果:

KNown: [<Tag a>]
KNown: [<Tag a>,<Tag b>]
KNown: [<Tag a>,<Tag b>,<Tag c>]
Result: [<Tag b>,<Tag c>]
------------------
KNown: [<Tag a>,<Tag c>]
Result: []

所以,不知何故,我不小心创建了一个闭包.据我所见,已知应该已经超出范围并在函数调用完成后消失.

如果我将chain_tags()的定义更改为未设置认值,则问题就会消失:

...
def tag_chain(tag,kNown):
...
o = tag_chain(a,[])
print "Result:",o
print "------------------"
o = tag_chain(a,o

为什么是这样?

解决方法

这是Python中常见的错误

def tag_chain(tag,kNown=[]):
  # ...

kNown = []并不意味着如果已知notpplied,则将其设为空列表;事实上,它与“匿名”列表相关联.每次知道认为该列表时,它都是相同的列表.

在这里做你想要的典型模式是:

def tag_chain(tag,kNown=None):
    if kNown is None:
        kNown = []
    # ...

如果未提供,则正确初始化已知的空列表.

相关文章

功能概要:(目前已实现功能)公共展示部分:1.网站首页展示...
大体上把Python中的数据类型分为如下几类: Number(数字) ...
开发之前第一步,就是构造整个的项目结构。这就好比作一幅画...
源码编译方式安装Apache首先下载Apache源码压缩包,地址为ht...
前面说完了此项目的创建及数据模型设计的过程。如果未看过,...
python中常用的写爬虫的库有urllib2、requests,对于大多数比...