我有一个类,其实例需要按照用户的指示格式化输出.有一种默认格式,可以覆盖.我这样实现了:
class A:
def __init__(self, params):
# ...
# by default printing all float values as percentages with 2 decimals
self.format_functions = {float: lambda x : '{:.2%}'.format(x)}
def __str__(self):
# uses self.format_functions to format output
# ...
a = A(params)
print(a) # uses default output formatting
# overriding default output formatting
# float printed as percentages 3 decimal digits; bool printed as Y / N
a.format_functions = {float : lambda x: '{:.3%}'.format(x),
bool : lambda x: 'Y' if x else 'N'}
print(a)
好吗?让我知道是否有更好的方法来设计它.
不幸的是,我需要挑选这个类的实例.但是只能在模块顶层定义的函数进行酸洗; lambda函数是不可取消的,所以我的format_functions实例属性打破了酸洗.
我尝试重写这个以使用类方法而不是lambda函数,但由于同样的原因仍然没有运气:
class A:
@classmethod
def default_float_format(cls, x):
return '{:.2%}'.format(x)
def __init__(self, params):
# ...
# by default printing all float values as percentages with 2 decimals
self.format_functions = {float: self.default_float_format}
def __str__(self):
# uses self.format_functions to format output
# ...
a = A(params)
pickle.dump(a) # Can't pickle <class 'method'>: attribute lookup builtins.method failed
请注意,即使我没有覆盖默认值,这里的酸洗也不起作用;只是我指定self.format_functions = {float:self.default_float_format}的事实打破了它.
该怎么办?我宁愿不污染命名空间并通过在模块级别定义default_float_format来破坏封装.
顺便说一下,为什么世界上泡菜会产生这种限制?对于最终用户来说,这无疑是一种无端和实质性的痛苦.
最佳答案 对于类实例或函数(以及方法)的pickle,Python的pickle取决于它们的名称可用作全局变量 – 字典中对方法的引用指向全局名称空间中不可用的名称 – 这是更好的说“模块命名空间” –
您可以通过自定义类的酸洗来创建teh“__setstate__”和“__getstate__”方法 – 但我认为您会更好,因为格式化函数不依赖于对象或类本身的任何信息(即使某些格式化函数有效,也可以将其作为参数传递,并在类范围之外定义一个函数.
这确实有效(Python 3.2):
def default_float_format( x):
return '{:.2%}'.format(x)
class A:
def __init__(self, params):
# ...
# by default printing all float values as percentages with 2 decimals
self.format_functions = {float: default_float_format}
def __str__(self):
# uses self.format_functions to format output
pass
a = A(1)
pickle.dumps(a)