我有一段代码如下:
tupvalue = [('html', 96), ('css', 115), ('map', 82)]
因此,当以所需格式为特定索引打印上述元组时,我发现了如下代码:
>>> '%s:%d' % tupvalue[0]
'html:96'
我想知道格式说明符’%s:%d’如何将单个值tupvalue [0]识别为两个值的元组吗?请通过文档参考说明此机制.
如何使用推导以所需的格式格式化tupvalue中的所有值,如所示示例所示?
解决方法:
首先,一个简单的问题:
How can I use a comprehension to format all the values in
tupvalue
in the required format as in the example shown?
这是一个列表理解:[‘%s:%d’%t对于tupvalue中的t]
现在,更难的问题!
how the single value
tupvalue[0]
is recognised as a tuple of two values by the format specifier'%s:%d'
?
您的直觉是这里发生了一些奇怪的事情,这是正确的.元组在使用字符串格式的语言中是特殊的.
>>> '%s:%d' % ('css', 115) # tuple is seen as two elements
'css:115'
>>> '%s:%d' % ['css', 115] # list is just seen as one object!
TypeError: not enough arguments for format string
百分比样式的字符串格式设置不正确.因此,如果您实际上想格式化一个元组,则必须将其包装在另一个元组中,这与任何其他类型的对象不同:
>>> '%s' % []
'[]'
>>> '%s' % ((),)
'()'
>>> '%s' % ()
TypeError: not enough arguments for format string
文档的相关部分位于4.7.2. printf
-style String Formatting节,其中提到:
If format requires a single argument, values may be a single non-tuple object. Otherwise, values must be a tuple with exactly the number of items specified by the format string
元组的奇怪处理是文档那部分开头的注释中提到的怪癖之一,也是推荐使用较新的字符串格式设置方法str.format的原因之一.
请注意,字符串格式的处理发生在运行时†.您可以使用抽象语法树对此进行验证:
>>> import ast
>>> ast.dump(ast.parse('"%s" % val'))
"Module(body=[Expr(value=BinOp(left=Str(s='%s'), op=Mod(), right=Name(id='val', ctx=Load())))])"
‘%s’%val解析为对’%s’和val的二进制操作,在CPython中,这是像BINARY_MODULO操作码那样,对str .__ mod __(val)进行处理.这意味着通常由str类型决定接收到的val不正确*时的处理方式*,这仅在计算表达式(即解释器到达该行)后才发生.因此,val是错误的类型还是元素太少/太多并不重要,这是运行时错误,而不是语法错误.
†在某些特殊情况下,CPython的窥孔优化器可以在编译时“恒定折叠”.
*除非val的类型为str的子类,否则type(val).__ rmod__ should be可以控制结果.