相关疑难解决方法(0)

"最小的惊讶"和可变的默认论证

任何修补Python足够长的人都被以下问题咬伤(或撕成碎片):

def foo(a=[]):
    a.append(5)
    return a
Run Code Online (Sandbox Code Playgroud)

Python新手希望这个函数总能返回一个只包含一个元素的列表:[5].结果却非常不同,而且非常惊人(对于新手来说):

>>> foo()
[5]
>>> foo()
[5, 5]
>>> foo()
[5, 5, 5]
>>> foo()
[5, 5, 5, 5]
>>> foo()
Run Code Online (Sandbox Code Playgroud)

我的一位经理曾经第一次遇到这个功能,并称其为该语言的"戏剧性设计缺陷".我回答说这个行为有一个潜在的解释,如果你不理解内部,那确实非常令人费解和意想不到.但是,我无法回答(对自己)以下问题:在函数定义中绑定默认参数的原因是什么,而不是在函数执行时?我怀疑经验丰富的行为有实际用途(谁真的在C中使用静态变量,没有繁殖错误?)

编辑:

巴泽克提出了一个有趣的例子.再加上你的大部分评论和特别是Utaal,我进一步阐述了:

>>> def a():
...     print("a executed")
...     return []
... 
>>>            
>>> def b(x=a()):
...     x.append(5)
...     print(x)
... 
a executed
>>> b()
[5]
>>> b()
[5, 5]
Run Code Online (Sandbox Code Playgroud)

对我而言,似乎设计决策是相对于放置参数范围的位置:在函数内部还是"与它一起"?

在函数内部进行绑定意味着在调用函数时x有效地绑定到指定的默认值,而不是定义,这会产生一个深层次的缺陷:def在某种意义上,该行将是"混合"的(部分绑定)函数对象)将在定义时发生,并在函数调用时发生部分(默认参数的赋值).

实际行为更加一致:执行该行时,该行的所有内容都会得到评估,这意味着在函数定义中.

python language-design least-astonishment default-parameters

2458
推荐指数
31
解决办法
14万
查看次数

有没有Python相当于Ruby的字符串插值?

Ruby示例:

name = "Spongebob Squarepants"
puts "Who lives in a Pineapple under the sea? \n#{name}."
Run Code Online (Sandbox Code Playgroud)

成功的Python字符串连接对我来说似乎很冗长.

python language-comparisons string-interpolation

328
推荐指数
5
解决办法
19万
查看次数

Python是否在Ruby中执行类似于"string#{var}"的变量插值?

在Python中,编写它是很繁琐的:

print "foo is" + bar + '.'
Run Code Online (Sandbox Code Playgroud)

我可以在Python中做这样的事吗?

print "foo is #{bar}."

ruby python language-comparisons string-formatting string-interpolation

34
推荐指数
5
解决办法
3万
查看次数

打印变量名称和内容作为调试工具; 寻找emacs/Python快捷方式

我发现自己经常添加调试"打印"语句 - 这样的东西:

print("a_variable_name: %s" % a_variable_name)
Run Code Online (Sandbox Code Playgroud)

你们都是那样做的?我是否在试图寻找优化方法的神经质?我可能正在研究一个函数并放入这些行中的大约六个,弄清楚它为什么不起作用,然后再将它们删除.

你有没有开发出有效的方法呢?

我在Emacs中编写Python代码.

python printing variables debugging emacs

11
推荐指数
1
解决办法
3114
查看次数

是一个字符串格式化程序,从其调用范围的不良做法中提取变量?

我有一些代码可以执行大量的字符串格式化.通常,我最终得到的代码如下:

"...".format(x=x, y=y, z=z, foo=foo, ...)
Run Code Online (Sandbox Code Playgroud)

我试图将大量变量插入到一个大字符串中.

是否有充分的理由不编写这样的函数来使用inspect模块来查找要插入的变量?

import inspect

def interpolate(s):
    return s.format(**inspect.currentframe().f_back.f_locals)

def generateTheString(x):
    y = foo(x)
    z = x + y
    # more calculations go here
    return interpolate("{x}, {y}, {z}")
Run Code Online (Sandbox Code Playgroud)

python

10
推荐指数
2
解决办法
1292
查看次数

我可以使用动态映射来解压缩Python中的关键字参数吗?

长话短说,我想用任意命名的参数调用格式,这将执行查找.

'{Thing1} and {other_thing}'.format(**my_mapping)
Run Code Online (Sandbox Code Playgroud)

我试过像这样实现my_mapping:

class Mapping(object):
  def __getitem__(self, key):
    return 'Proxied: %s' % key
my_mapping = Mapping()
Run Code Online (Sandbox Code Playgroud)

这在调用时按预期工作my_mapping['anything'].但是当传递给如上所示的format()时,我得到:

TypeError: format() argument after ** must be a mapping, not Mapping
Run Code Online (Sandbox Code Playgroud)

我尝试了子类dict而不是object,但现在调用format()如图所示KeyError.我甚至实现__contains__return True,但仍然KeyError.

所以它似乎**不只是调用__getitem__传入的对象.有谁知道如何解决这个问题?

python keyword-argument

6
推荐指数
2
解决办法
2377
查看次数