给定一个格式字符串字典,我想做级联/递归字符串插值。
FOLDERS = dict(home="/home/user",
workspace="{home}/workspace",
app_project="{workspace}/{app_name}",
app_name="my_app")
我从这个实现开始:
def interpolate(attrs):
remain = [k for k, v in attrs.items() if "{" in v]
while remain:
for k in remain:
attrs[k] = attrs[k].format(**attrs)
remain = [k for k in remain if "{" in attrs[k]]
该interpolate()
函数首先选择格式字符串。然后,它替换字符串,直到不再有格式字符串。
当我使用以下 Python 字典调用此函数时,我得到:
>>> import pprint
>>> pprint.pprint(FOLDERS)
{'app_name': 'my_app',
'app_project': '/home/user/workspace/my_app',
'home': '/home/user',
'workspace': '/home/user/workspace'}
结果没问题,但是这个实现没有检测到参考周期。
例如,以下调用导致无限循环!
>>> interpolate({'home': '{home}'})
谁能给我一个更好的实现?
编辑:解决方案
我认为 Leon 的解决方案既好又简单,Serge Bellesta 也是如此。
我会这样实现它:
def interpolate(attrs):
remain = [k for k, v in attrs.items() if "{" in v]
while remain:
for k in remain:
attrs[k] = attrs[k].format(**attrs)
fmt = '{' + k + '}'
if fmt in attrs[k]: # check for reference cycles
raise ValueError("Reference cycle found for '{k}'!".format(k=k))
remain = [k for k in remain if "{" in attrs[k]]