我想使用包含点的键的字典 parDict
并发现 update
函数不会解释带点的键,尽管字典工作正常。
点符号是由于参数集的面向对象。
下面的例子说明了“不一致”。
parDict = {}
parDict['a'] = 1
parDict['b'] = 2
parDict['group1.b'] = 3
parDict
的多次更新可以在一个对我很重要的命令中完成
parDict.update(a=4, b=4)
但无法识别以下更新
parDict.update(group1.b=4)
我得到:“语法错误:表达式不能包含赋值,...”
然而,
parDict['group1.b'] = 4
工作正常。
有没有办法解决这种“不一致”问题,即使对于名称中带有点的键,也可以使用 update()
?
了解更广泛的背景为什么 update()
在这里不起作用会很有趣。
答案 0 :(得分:4)
首先,让我们看看如何调用 update
。根据{{3}}(强调我的),
update()
接受另一个字典对象或键/值对的可迭代对象(作为元组或其他长度为 2 的可迭代对象)。 如果指定了关键字参数,则会使用这些键/值对更新字典:d.update(red=1, blue=2)
。
但是如何定义关键字参数呢?根据{{3}}(强调我的),
<块引用>关键字参数:在函数调用中以标识符开头的参数(例如name=
)或在以**开头的字典中作为值传递.
但什么是标识符?根据{{3}},
<块引用>标识符的有效字符与 Python 2.x 中的相同:大写和小写字母 A
到 Z
、下划线 _
以及除第一个字符外,数字 0
到 9
。
哦,好的。因此,您可以使用 name
或 a
等标识符,但不能使用 group1.b
等标识符,因为其中包含一个点。
回到update
方法,如果你有一个不是标识符的字典键,你可以使用字典来更新它:
parDict.update({"group1.b": 4})
答案 1 :(得分:1)
我可以想到以下解决方法。通过这种方式,您可以使用单个命令持续更新 parDict
。
parDict = {}
parDict['a'] = 1
parDict['b'] = 2
parDict['group1.b'] = 3
parDict.update(a=4, b=4)
print(parDict)
parDict.update({"a":5, "group1.b":7})
print(parDict)
输出:
{'a': 4, 'b': 4, 'group1.b': 3}
{'a': 5, 'b': 4, 'group1.b': 7}
答案 2 :(得分:0)
我很高兴收到有关 parDict 问题的意见,尽管我最初忽略了“键”和“标识符”之间的区别是非常基本的。我想到的目的是通过面向对象的参数结构简化命令行交互。这是一个具有一定普遍性的问题,也许这里有比我在下面建议的更好的解决方案?
将 update() 与元组一起使用很有吸引力,更具可读性,并且避免使用@wjandrea 发布的链接中指出的一些标志。 但是要以这种方式使用它,我们需要引入另一个字典,即我们有 parDict 具有短的唯一参数名称并使用标识符和对应的值,然后引入 parLocation,这是一个将短名称参数名称与面向对象的位置相关联的字典字符串。
解决办法
parDict = {}
parDict['a'] = 1
parDict['b'] = 2
parDict['group1_b'] = 3
和
parLocation = {}
parLocation['a'] = 'a'
parLocation['b'] = 'b'
parLocation['group1_b'] = 'group1.b'
对于命令行交互,我现在可以编写
parDict.update(b=4, group1_b=4)
对于参数值被带到面向对象系统的内部处理,我写了一些类似的东西
for key in parDict.keys(): set(parLocation[key], parDict[key])
其中 set() 是一些将参数“位置”和“值”作为参数的函数。
由于问题具有一定的普遍性,我认为这里可能有其他更好或更直接的方法吗?